virtio_net: add link status handling
[linux-2.6] / drivers / net / skfp / rmt.c
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