diff options
Diffstat (limited to 'drivers/net/skfp/rmt.c')
-rw-r--r-- | drivers/net/skfp/rmt.c | 654 |
1 files changed, 654 insertions, 0 deletions
diff --git a/drivers/net/skfp/rmt.c b/drivers/net/skfp/rmt.c new file mode 100644 index 000000000000..ef8d5672d9e8 --- /dev/null +++ b/drivers/net/skfp/rmt.c | |||
@@ -0,0 +1,654 @@ | |||
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 RMT | ||
19 | Ring Management | ||
20 | */ | ||
21 | |||
22 | /* | ||
23 | * Hardware independent state machine implemantation | ||
24 | * The following external SMT functions are referenced : | ||
25 | * | ||
26 | * queue_event() | ||
27 | * smt_timer_start() | ||
28 | * smt_timer_stop() | ||
29 | * | ||
30 | * The following external HW dependent functions are referenced : | ||
31 | * sm_ma_control() | ||
32 | * sm_mac_check_beacon_claim() | ||
33 | * | ||
34 | * The following HW dependent events are required : | ||
35 | * RM_RING_OP | ||
36 | * RM_RING_NON_OP | ||
37 | * RM_MY_BEACON | ||
38 | * RM_OTHER_BEACON | ||
39 | * RM_MY_CLAIM | ||
40 | * RM_TRT_EXP | ||
41 | * RM_VALID_CLAIM | ||
42 | * | ||
43 | */ | ||
44 | |||
45 | #include "h/types.h" | ||
46 | #include "h/fddi.h" | ||
47 | #include "h/smc.h" | ||
48 | |||
49 | #define KERNEL | ||
50 | #include "h/smtstate.h" | ||
51 | |||
52 | #ifndef lint | ||
53 | static const char ID_sccs[] = "@(#)rmt.c 2.13 99/07/02 (C) SK " ; | ||
54 | #endif | ||
55 | |||
56 | /* | ||
57 | * FSM Macros | ||
58 | */ | ||
59 | #define AFLAG 0x10 | ||
60 | #define GO_STATE(x) (smc->mib.m[MAC0].fddiMACRMTState = (x)|AFLAG) | ||
61 | #define ACTIONS_DONE() (smc->mib.m[MAC0].fddiMACRMTState &= ~AFLAG) | ||
62 | #define ACTIONS(x) (x|AFLAG) | ||
63 | |||
64 | #define RM0_ISOLATED 0 | ||
65 | #define RM1_NON_OP 1 /* not operational */ | ||
66 | #define RM2_RING_OP 2 /* ring operational */ | ||
67 | #define RM3_DETECT 3 /* detect dupl addresses */ | ||
68 | #define RM4_NON_OP_DUP 4 /* dupl. addr detected */ | ||
69 | #define RM5_RING_OP_DUP 5 /* ring oper. with dupl. addr */ | ||
70 | #define RM6_DIRECTED 6 /* sending directed beacons */ | ||
71 | #define RM7_TRACE 7 /* trace initiated */ | ||
72 | |||
73 | #ifdef DEBUG | ||
74 | /* | ||
75 | * symbolic state names | ||
76 | */ | ||
77 | static const char * const rmt_states[] = { | ||
78 | "RM0_ISOLATED","RM1_NON_OP","RM2_RING_OP","RM3_DETECT", | ||
79 | "RM4_NON_OP_DUP","RM5_RING_OP_DUP","RM6_DIRECTED", | ||
80 | "RM7_TRACE" | ||
81 | } ; | ||
82 | |||
83 | /* | ||
84 | * symbolic event names | ||
85 | */ | ||
86 | static const char * const rmt_events[] = { | ||
87 | "NONE","RM_RING_OP","RM_RING_NON_OP","RM_MY_BEACON", | ||
88 | "RM_OTHER_BEACON","RM_MY_CLAIM","RM_TRT_EXP","RM_VALID_CLAIM", | ||
89 | "RM_JOIN","RM_LOOP","RM_DUP_ADDR","RM_ENABLE_FLAG", | ||
90 | "RM_TIMEOUT_NON_OP","RM_TIMEOUT_T_STUCK", | ||
91 | "RM_TIMEOUT_ANNOUNCE","RM_TIMEOUT_T_DIRECT", | ||
92 | "RM_TIMEOUT_D_MAX","RM_TIMEOUT_POLL","RM_TX_STATE_CHANGE" | ||
93 | } ; | ||
94 | #endif | ||
95 | |||
96 | /* | ||
97 | * Globals | ||
98 | * in struct s_rmt | ||
99 | */ | ||
100 | |||
101 | |||
102 | /* | ||
103 | * function declarations | ||
104 | */ | ||
105 | static void rmt_fsm(struct s_smc *smc, int cmd); | ||
106 | static void start_rmt_timer0(struct s_smc *smc, u_long value, int event); | ||
107 | static void start_rmt_timer1(struct s_smc *smc, u_long value, int event); | ||
108 | static void start_rmt_timer2(struct s_smc *smc, u_long value, int event); | ||
109 | static void stop_rmt_timer0(struct s_smc *smc); | ||
110 | static void stop_rmt_timer1(struct s_smc *smc); | ||
111 | static void stop_rmt_timer2(struct s_smc *smc); | ||
112 | static void rmt_dup_actions(struct s_smc *smc); | ||
113 | static void rmt_reinsert_actions(struct s_smc *smc); | ||
114 | static void rmt_leave_actions(struct s_smc *smc); | ||
115 | static void rmt_new_dup_actions(struct s_smc *smc); | ||
116 | |||
117 | #ifndef SUPERNET_3 | ||
118 | extern void restart_trt_for_dbcn() ; | ||
119 | #endif /*SUPERNET_3*/ | ||
120 | |||
121 | /* | ||
122 | init RMT state machine | ||
123 | clear all RMT vars and flags | ||
124 | */ | ||
125 | void rmt_init(struct s_smc *smc) | ||
126 | { | ||
127 | smc->mib.m[MAC0].fddiMACRMTState = ACTIONS(RM0_ISOLATED) ; | ||
128 | smc->r.dup_addr_test = DA_NONE ; | ||
129 | smc->r.da_flag = 0 ; | ||
130 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; | ||
131 | smc->r.sm_ma_avail = FALSE ; | ||
132 | smc->r.loop_avail = 0 ; | ||
133 | smc->r.bn_flag = 0 ; | ||
134 | smc->r.jm_flag = 0 ; | ||
135 | smc->r.no_flag = TRUE ; | ||
136 | } | ||
137 | |||
138 | /* | ||
139 | RMT state machine | ||
140 | called by dispatcher | ||
141 | |||
142 | do | ||
143 | display state change | ||
144 | process event | ||
145 | until SM is stable | ||
146 | */ | ||
147 | void rmt(struct s_smc *smc, int event) | ||
148 | { | ||
149 | int state ; | ||
150 | |||
151 | do { | ||
152 | DB_RMT("RMT : state %s%s", | ||
153 | (smc->mib.m[MAC0].fddiMACRMTState & AFLAG) ? "ACTIONS " : "", | ||
154 | rmt_states[smc->mib.m[MAC0].fddiMACRMTState & ~AFLAG]) ; | ||
155 | DB_RMT(" event %s\n",rmt_events[event],0) ; | ||
156 | state = smc->mib.m[MAC0].fddiMACRMTState ; | ||
157 | rmt_fsm(smc,event) ; | ||
158 | event = 0 ; | ||
159 | } while (state != smc->mib.m[MAC0].fddiMACRMTState) ; | ||
160 | rmt_state_change(smc,(int)smc->mib.m[MAC0].fddiMACRMTState) ; | ||
161 | } | ||
162 | |||
163 | /* | ||
164 | process RMT event | ||
165 | */ | ||
166 | static void rmt_fsm(struct s_smc *smc, int cmd) | ||
167 | { | ||
168 | /* | ||
169 | * RM00-RM70 : from all states | ||
170 | */ | ||
171 | if (!smc->r.rm_join && !smc->r.rm_loop && | ||
172 | smc->mib.m[MAC0].fddiMACRMTState != ACTIONS(RM0_ISOLATED) && | ||
173 | smc->mib.m[MAC0].fddiMACRMTState != RM0_ISOLATED) { | ||
174 | RS_SET(smc,RS_NORINGOP) ; | ||
175 | rmt_indication(smc,0) ; | ||
176 | GO_STATE(RM0_ISOLATED) ; | ||
177 | return ; | ||
178 | } | ||
179 | |||
180 | switch(smc->mib.m[MAC0].fddiMACRMTState) { | ||
181 | case ACTIONS(RM0_ISOLATED) : | ||
182 | stop_rmt_timer0(smc) ; | ||
183 | stop_rmt_timer1(smc) ; | ||
184 | stop_rmt_timer2(smc) ; | ||
185 | |||
186 | /* | ||
187 | * Disable MAC. | ||
188 | */ | ||
189 | sm_ma_control(smc,MA_OFFLINE) ; | ||
190 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; | ||
191 | smc->r.loop_avail = FALSE ; | ||
192 | smc->r.sm_ma_avail = FALSE ; | ||
193 | smc->r.no_flag = TRUE ; | ||
194 | DB_RMTN(1,"RMT : ISOLATED\n",0,0) ; | ||
195 | ACTIONS_DONE() ; | ||
196 | break ; | ||
197 | case RM0_ISOLATED : | ||
198 | /*RM01*/ | ||
199 | if (smc->r.rm_join || smc->r.rm_loop) { | ||
200 | /* | ||
201 | * According to the standard the MAC must be reset | ||
202 | * here. The FORMAC will be initialized and Claim | ||
203 | * and Beacon Frames will be uploaded to the MAC. | ||
204 | * So any change of Treq will take effect NOW. | ||
205 | */ | ||
206 | sm_ma_control(smc,MA_RESET) ; | ||
207 | GO_STATE(RM1_NON_OP) ; | ||
208 | break ; | ||
209 | } | ||
210 | break ; | ||
211 | case ACTIONS(RM1_NON_OP) : | ||
212 | start_rmt_timer0(smc,smc->s.rmt_t_non_op,RM_TIMEOUT_NON_OP) ; | ||
213 | stop_rmt_timer1(smc) ; | ||
214 | stop_rmt_timer2(smc) ; | ||
215 | sm_ma_control(smc,MA_BEACON) ; | ||
216 | DB_RMTN(1,"RMT : RING DOWN\n",0,0) ; | ||
217 | RS_SET(smc,RS_NORINGOP) ; | ||
218 | smc->r.sm_ma_avail = FALSE ; | ||
219 | rmt_indication(smc,0) ; | ||
220 | ACTIONS_DONE() ; | ||
221 | break ; | ||
222 | case RM1_NON_OP : | ||
223 | /*RM12*/ | ||
224 | if (cmd == RM_RING_OP) { | ||
225 | RS_SET(smc,RS_RINGOPCHANGE) ; | ||
226 | GO_STATE(RM2_RING_OP) ; | ||
227 | break ; | ||
228 | } | ||
229 | /*RM13*/ | ||
230 | else if (cmd == RM_TIMEOUT_NON_OP) { | ||
231 | smc->r.bn_flag = FALSE ; | ||
232 | smc->r.no_flag = TRUE ; | ||
233 | GO_STATE(RM3_DETECT) ; | ||
234 | break ; | ||
235 | } | ||
236 | break ; | ||
237 | case ACTIONS(RM2_RING_OP) : | ||
238 | stop_rmt_timer0(smc) ; | ||
239 | stop_rmt_timer1(smc) ; | ||
240 | stop_rmt_timer2(smc) ; | ||
241 | smc->r.no_flag = FALSE ; | ||
242 | if (smc->r.rm_loop) | ||
243 | smc->r.loop_avail = TRUE ; | ||
244 | if (smc->r.rm_join) { | ||
245 | smc->r.sm_ma_avail = TRUE ; | ||
246 | if (smc->mib.m[MAC0].fddiMACMA_UnitdataEnable) | ||
247 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = TRUE ; | ||
248 | else | ||
249 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; | ||
250 | } | ||
251 | DB_RMTN(1,"RMT : RING UP\n",0,0) ; | ||
252 | RS_CLEAR(smc,RS_NORINGOP) ; | ||
253 | RS_SET(smc,RS_RINGOPCHANGE) ; | ||
254 | rmt_indication(smc,1) ; | ||
255 | smt_stat_counter(smc,0) ; | ||
256 | ACTIONS_DONE() ; | ||
257 | break ; | ||
258 | case RM2_RING_OP : | ||
259 | /*RM21*/ | ||
260 | if (cmd == RM_RING_NON_OP) { | ||
261 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; | ||
262 | smc->r.loop_avail = FALSE ; | ||
263 | RS_SET(smc,RS_RINGOPCHANGE) ; | ||
264 | GO_STATE(RM1_NON_OP) ; | ||
265 | break ; | ||
266 | } | ||
267 | /*RM22a*/ | ||
268 | else if (cmd == RM_ENABLE_FLAG) { | ||
269 | if (smc->mib.m[MAC0].fddiMACMA_UnitdataEnable) | ||
270 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = TRUE ; | ||
271 | else | ||
272 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; | ||
273 | } | ||
274 | /*RM25*/ | ||
275 | else if (smc->r.dup_addr_test == DA_FAILED) { | ||
276 | smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; | ||
277 | smc->r.loop_avail = FALSE ; | ||
278 | smc->r.da_flag = TRUE ; | ||
279 | GO_STATE(RM5_RING_OP_DUP) ; | ||
280 | break ; | ||
281 | } | ||
282 | break ; | ||
283 | case ACTIONS(RM3_DETECT) : | ||
284 | start_rmt_timer0(smc,smc->s.mac_d_max*2,RM_TIMEOUT_D_MAX) ; | ||
285 | start_rmt_timer1(smc,smc->s.rmt_t_stuck,RM_TIMEOUT_T_STUCK) ; | ||
286 | start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL) ; | ||
287 | sm_mac_check_beacon_claim(smc) ; | ||
288 | DB_RMTN(1,"RMT : RM3_DETECT\n",0,0) ; | ||
289 | ACTIONS_DONE() ; | ||
290 | break ; | ||
291 | case RM3_DETECT : | ||
292 | if (cmd == RM_TIMEOUT_POLL) { | ||
293 | start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL); | ||
294 | sm_mac_check_beacon_claim(smc) ; | ||
295 | break ; | ||
296 | } | ||
297 | if (cmd == RM_TIMEOUT_D_MAX) { | ||
298 | smc->r.timer0_exp = TRUE ; | ||
299 | } | ||
300 | /* | ||
301 | *jd(22-Feb-1999) | ||
302 | * We need a time ">= 2*mac_d_max" since we had finished | ||
303 | * Claim or Beacon state. So we will restart timer0 at | ||
304 | * every state change. | ||
305 | */ | ||
306 | if (cmd == RM_TX_STATE_CHANGE) { | ||
307 | start_rmt_timer0(smc, | ||
308 | smc->s.mac_d_max*2, | ||
309 | RM_TIMEOUT_D_MAX) ; | ||
310 | } | ||
311 | /*RM32*/ | ||
312 | if (cmd == RM_RING_OP) { | ||
313 | GO_STATE(RM2_RING_OP) ; | ||
314 | break ; | ||
315 | } | ||
316 | /*RM33a*/ | ||
317 | else if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) | ||
318 | && smc->r.bn_flag) { | ||
319 | smc->r.bn_flag = FALSE ; | ||
320 | } | ||
321 | /*RM33b*/ | ||
322 | else if (cmd == RM_TRT_EXP && !smc->r.bn_flag) { | ||
323 | int tx ; | ||
324 | /* | ||
325 | * set bn_flag only if in state T4 or T5: | ||
326 | * only if we're the beaconer should we start the | ||
327 | * trace ! | ||
328 | */ | ||
329 | if ((tx = sm_mac_get_tx_state(smc)) == 4 || tx == 5) { | ||
330 | DB_RMTN(2,"RMT : DETECT && TRT_EXPIRED && T4/T5\n",0,0); | ||
331 | smc->r.bn_flag = TRUE ; | ||
332 | /* | ||
333 | * If one of the upstream stations beaconed | ||
334 | * and the link to the upstream neighbor is | ||
335 | * lost we need to restart the stuck timer to | ||
336 | * check the "stuck beacon" condition. | ||
337 | */ | ||
338 | start_rmt_timer1(smc,smc->s.rmt_t_stuck, | ||
339 | RM_TIMEOUT_T_STUCK) ; | ||
340 | } | ||
341 | /* | ||
342 | * We do NOT need to clear smc->r.bn_flag in case of | ||
343 | * not being in state T4 or T5, because the flag | ||
344 | * must be cleared in order to get in this condition. | ||
345 | */ | ||
346 | |||
347 | DB_RMTN(2, | ||
348 | "RMT : sm_mac_get_tx_state() = %d (bn_flag = %d)\n", | ||
349 | tx,smc->r.bn_flag) ; | ||
350 | } | ||
351 | /*RM34a*/ | ||
352 | else if (cmd == RM_MY_CLAIM && smc->r.timer0_exp) { | ||
353 | rmt_new_dup_actions(smc) ; | ||
354 | GO_STATE(RM4_NON_OP_DUP) ; | ||
355 | break ; | ||
356 | } | ||
357 | /*RM34b*/ | ||
358 | else if (cmd == RM_MY_BEACON && smc->r.timer0_exp) { | ||
359 | rmt_new_dup_actions(smc) ; | ||
360 | GO_STATE(RM4_NON_OP_DUP) ; | ||
361 | break ; | ||
362 | } | ||
363 | /*RM34c*/ | ||
364 | else if (cmd == RM_VALID_CLAIM) { | ||
365 | rmt_new_dup_actions(smc) ; | ||
366 | GO_STATE(RM4_NON_OP_DUP) ; | ||
367 | break ; | ||
368 | } | ||
369 | /*RM36*/ | ||
370 | else if (cmd == RM_TIMEOUT_T_STUCK && | ||
371 | smc->r.rm_join && smc->r.bn_flag) { | ||
372 | GO_STATE(RM6_DIRECTED) ; | ||
373 | break ; | ||
374 | } | ||
375 | break ; | ||
376 | case ACTIONS(RM4_NON_OP_DUP) : | ||
377 | start_rmt_timer0(smc,smc->s.rmt_t_announce,RM_TIMEOUT_ANNOUNCE); | ||
378 | start_rmt_timer1(smc,smc->s.rmt_t_stuck,RM_TIMEOUT_T_STUCK) ; | ||
379 | start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL) ; | ||
380 | sm_mac_check_beacon_claim(smc) ; | ||
381 | DB_RMTN(1,"RMT : RM4_NON_OP_DUP\n",0,0) ; | ||
382 | ACTIONS_DONE() ; | ||
383 | break ; | ||
384 | case RM4_NON_OP_DUP : | ||
385 | if (cmd == RM_TIMEOUT_POLL) { | ||
386 | start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL); | ||
387 | sm_mac_check_beacon_claim(smc) ; | ||
388 | break ; | ||
389 | } | ||
390 | /*RM41*/ | ||
391 | if (!smc->r.da_flag) { | ||
392 | GO_STATE(RM1_NON_OP) ; | ||
393 | break ; | ||
394 | } | ||
395 | /*RM44a*/ | ||
396 | else if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) && | ||
397 | smc->r.bn_flag) { | ||
398 | smc->r.bn_flag = FALSE ; | ||
399 | } | ||
400 | /*RM44b*/ | ||
401 | else if (cmd == RM_TRT_EXP && !smc->r.bn_flag) { | ||
402 | int tx ; | ||
403 | /* | ||
404 | * set bn_flag only if in state T4 or T5: | ||
405 | * only if we're the beaconer should we start the | ||
406 | * trace ! | ||
407 | */ | ||
408 | if ((tx = sm_mac_get_tx_state(smc)) == 4 || tx == 5) { | ||
409 | DB_RMTN(2,"RMT : NOPDUP && TRT_EXPIRED && T4/T5\n",0,0); | ||
410 | smc->r.bn_flag = TRUE ; | ||
411 | /* | ||
412 | * If one of the upstream stations beaconed | ||
413 | * and the link to the upstream neighbor is | ||
414 | * lost we need to restart the stuck timer to | ||
415 | * check the "stuck beacon" condition. | ||
416 | */ | ||
417 | start_rmt_timer1(smc,smc->s.rmt_t_stuck, | ||
418 | RM_TIMEOUT_T_STUCK) ; | ||
419 | } | ||
420 | /* | ||
421 | * We do NOT need to clear smc->r.bn_flag in case of | ||
422 | * not being in state T4 or T5, because the flag | ||
423 | * must be cleared in order to get in this condition. | ||
424 | */ | ||
425 | |||
426 | DB_RMTN(2, | ||
427 | "RMT : sm_mac_get_tx_state() = %d (bn_flag = %d)\n", | ||
428 | tx,smc->r.bn_flag) ; | ||
429 | } | ||
430 | /*RM44c*/ | ||
431 | else if (cmd == RM_TIMEOUT_ANNOUNCE && !smc->r.bn_flag) { | ||
432 | rmt_dup_actions(smc) ; | ||
433 | } | ||
434 | /*RM45*/ | ||
435 | else if (cmd == RM_RING_OP) { | ||
436 | smc->r.no_flag = FALSE ; | ||
437 | GO_STATE(RM5_RING_OP_DUP) ; | ||
438 | break ; | ||
439 | } | ||
440 | /*RM46*/ | ||
441 | else if (cmd == RM_TIMEOUT_T_STUCK && | ||
442 | smc->r.rm_join && smc->r.bn_flag) { | ||
443 | GO_STATE(RM6_DIRECTED) ; | ||
444 | break ; | ||
445 | } | ||
446 | break ; | ||
447 | case ACTIONS(RM5_RING_OP_DUP) : | ||
448 | stop_rmt_timer0(smc) ; | ||
449 | stop_rmt_timer1(smc) ; | ||
450 | stop_rmt_timer2(smc) ; | ||
451 | DB_RMTN(1,"RMT : RM5_RING_OP_DUP\n",0,0) ; | ||
452 | ACTIONS_DONE() ; | ||
453 | break; | ||
454 | case RM5_RING_OP_DUP : | ||
455 | /*RM52*/ | ||
456 | if (smc->r.dup_addr_test == DA_PASSED) { | ||
457 | smc->r.da_flag = FALSE ; | ||
458 | GO_STATE(RM2_RING_OP) ; | ||
459 | break ; | ||
460 | } | ||
461 | /*RM54*/ | ||
462 | else if (cmd == RM_RING_NON_OP) { | ||
463 | smc->r.jm_flag = FALSE ; | ||
464 | smc->r.bn_flag = FALSE ; | ||
465 | GO_STATE(RM4_NON_OP_DUP) ; | ||
466 | break ; | ||
467 | } | ||
468 | break ; | ||
469 | case ACTIONS(RM6_DIRECTED) : | ||
470 | start_rmt_timer0(smc,smc->s.rmt_t_direct,RM_TIMEOUT_T_DIRECT) ; | ||
471 | stop_rmt_timer1(smc) ; | ||
472 | start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL) ; | ||
473 | sm_ma_control(smc,MA_DIRECTED) ; | ||
474 | RS_SET(smc,RS_BEACON) ; | ||
475 | DB_RMTN(1,"RMT : RM6_DIRECTED\n",0,0) ; | ||
476 | ACTIONS_DONE() ; | ||
477 | break ; | ||
478 | case RM6_DIRECTED : | ||
479 | /*RM63*/ | ||
480 | if (cmd == RM_TIMEOUT_POLL) { | ||
481 | start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL); | ||
482 | sm_mac_check_beacon_claim(smc) ; | ||
483 | #ifndef SUPERNET_3 | ||
484 | /* Because of problems with the Supernet II chip set | ||
485 | * sending of Directed Beacon will stop after 165ms | ||
486 | * therefore restart_trt_for_dbcn(smc) will be called | ||
487 | * to prevent this. | ||
488 | */ | ||
489 | restart_trt_for_dbcn(smc) ; | ||
490 | #endif /*SUPERNET_3*/ | ||
491 | break ; | ||
492 | } | ||
493 | if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) && | ||
494 | !smc->r.da_flag) { | ||
495 | smc->r.bn_flag = FALSE ; | ||
496 | GO_STATE(RM3_DETECT) ; | ||
497 | break ; | ||
498 | } | ||
499 | /*RM64*/ | ||
500 | else if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) && | ||
501 | smc->r.da_flag) { | ||
502 | smc->r.bn_flag = FALSE ; | ||
503 | GO_STATE(RM4_NON_OP_DUP) ; | ||
504 | break ; | ||
505 | } | ||
506 | /*RM67*/ | ||
507 | else if (cmd == RM_TIMEOUT_T_DIRECT) { | ||
508 | GO_STATE(RM7_TRACE) ; | ||
509 | break ; | ||
510 | } | ||
511 | break ; | ||
512 | case ACTIONS(RM7_TRACE) : | ||
513 | stop_rmt_timer0(smc) ; | ||
514 | stop_rmt_timer1(smc) ; | ||
515 | stop_rmt_timer2(smc) ; | ||
516 | smc->e.trace_prop |= ENTITY_BIT(ENTITY_MAC) ; | ||
517 | queue_event(smc,EVENT_ECM,EC_TRACE_PROP) ; | ||
518 | DB_RMTN(1,"RMT : RM7_TRACE\n",0,0) ; | ||
519 | ACTIONS_DONE() ; | ||
520 | break ; | ||
521 | case RM7_TRACE : | ||
522 | break ; | ||
523 | default: | ||
524 | SMT_PANIC(smc,SMT_E0122, SMT_E0122_MSG) ; | ||
525 | break; | ||
526 | } | ||
527 | } | ||
528 | |||
529 | /* | ||
530 | * (jd) RMT duplicate address actions | ||
531 | * leave the ring or reinsert just as configured | ||
532 | */ | ||
533 | static void rmt_dup_actions(struct s_smc *smc) | ||
534 | { | ||
535 | if (smc->r.jm_flag) { | ||
536 | } | ||
537 | else { | ||
538 | if (smc->s.rmt_dup_mac_behavior) { | ||
539 | SMT_ERR_LOG(smc,SMT_E0138, SMT_E0138_MSG) ; | ||
540 | rmt_reinsert_actions(smc) ; | ||
541 | } | ||
542 | else { | ||
543 | SMT_ERR_LOG(smc,SMT_E0135, SMT_E0135_MSG) ; | ||
544 | rmt_leave_actions(smc) ; | ||
545 | } | ||
546 | } | ||
547 | } | ||
548 | |||
549 | /* | ||
550 | * Reconnect to the Ring | ||
551 | */ | ||
552 | static void rmt_reinsert_actions(struct s_smc *smc) | ||
553 | { | ||
554 | queue_event(smc,EVENT_ECM,EC_DISCONNECT) ; | ||
555 | queue_event(smc,EVENT_ECM,EC_CONNECT) ; | ||
556 | } | ||
557 | |||
558 | /* | ||
559 | * duplicate address detected | ||
560 | */ | ||
561 | static void rmt_new_dup_actions(struct s_smc *smc) | ||
562 | { | ||
563 | smc->r.da_flag = TRUE ; | ||
564 | smc->r.bn_flag = FALSE ; | ||
565 | smc->r.jm_flag = FALSE ; | ||
566 | /* | ||
567 | * we have three options : change address, jam or leave | ||
568 | * we leave the ring as default | ||
569 | * Optionally it's possible to reinsert after leaving the Ring | ||
570 | * but this will not conform with SMT Spec. | ||
571 | */ | ||
572 | if (smc->s.rmt_dup_mac_behavior) { | ||
573 | SMT_ERR_LOG(smc,SMT_E0138, SMT_E0138_MSG) ; | ||
574 | rmt_reinsert_actions(smc) ; | ||
575 | } | ||
576 | else { | ||
577 | SMT_ERR_LOG(smc,SMT_E0135, SMT_E0135_MSG) ; | ||
578 | rmt_leave_actions(smc) ; | ||
579 | } | ||
580 | } | ||
581 | |||
582 | |||
583 | /* | ||
584 | * leave the ring | ||
585 | */ | ||
586 | static void rmt_leave_actions(struct s_smc *smc) | ||
587 | { | ||
588 | queue_event(smc,EVENT_ECM,EC_DISCONNECT) ; | ||
589 | /* | ||
590 | * Note: Do NOT try again later. (with please reconnect) | ||
591 | * The station must be left from the ring! | ||
592 | */ | ||
593 | } | ||
594 | |||
595 | /* | ||
596 | * SMT timer interface | ||
597 | * start RMT timer 0 | ||
598 | */ | ||
599 | static void start_rmt_timer0(struct s_smc *smc, u_long value, int event) | ||
600 | { | ||
601 | smc->r.timer0_exp = FALSE ; /* clear timer event flag */ | ||
602 | smt_timer_start(smc,&smc->r.rmt_timer0,value,EV_TOKEN(EVENT_RMT,event)); | ||
603 | } | ||
604 | |||
605 | /* | ||
606 | * SMT timer interface | ||
607 | * start RMT timer 1 | ||
608 | */ | ||
609 | static void start_rmt_timer1(struct s_smc *smc, u_long value, int event) | ||
610 | { | ||
611 | smc->r.timer1_exp = FALSE ; /* clear timer event flag */ | ||
612 | smt_timer_start(smc,&smc->r.rmt_timer1,value,EV_TOKEN(EVENT_RMT,event)); | ||
613 | } | ||
614 | |||
615 | /* | ||
616 | * SMT timer interface | ||
617 | * start RMT timer 2 | ||
618 | */ | ||
619 | static void start_rmt_timer2(struct s_smc *smc, u_long value, int event) | ||
620 | { | ||
621 | smc->r.timer2_exp = FALSE ; /* clear timer event flag */ | ||
622 | smt_timer_start(smc,&smc->r.rmt_timer2,value,EV_TOKEN(EVENT_RMT,event)); | ||
623 | } | ||
624 | |||
625 | /* | ||
626 | * SMT timer interface | ||
627 | * stop RMT timer 0 | ||
628 | */ | ||
629 | static void stop_rmt_timer0(struct s_smc *smc) | ||
630 | { | ||
631 | if (smc->r.rmt_timer0.tm_active) | ||
632 | smt_timer_stop(smc,&smc->r.rmt_timer0) ; | ||
633 | } | ||
634 | |||
635 | /* | ||
636 | * SMT timer interface | ||
637 | * stop RMT timer 1 | ||
638 | */ | ||
639 | static void stop_rmt_timer1(struct s_smc *smc) | ||
640 | { | ||
641 | if (smc->r.rmt_timer1.tm_active) | ||
642 | smt_timer_stop(smc,&smc->r.rmt_timer1) ; | ||
643 | } | ||
644 | |||
645 | /* | ||
646 | * SMT timer interface | ||
647 | * stop RMT timer 2 | ||
648 | */ | ||
649 | static void stop_rmt_timer2(struct s_smc *smc) | ||
650 | { | ||
651 | if (smc->r.rmt_timer2.tm_active) | ||
652 | smt_timer_stop(smc,&smc->r.rmt_timer2) ; | ||
653 | } | ||
654 | |||