OpenCores
URL https://opencores.org/ocsvn/test_project/test_project/trunk

Subversion Repositories test_project

[/] [test_project/] [trunk/] [linux_sd_driver/] [drivers/] [net/] [skfp/] [ecm.c] - Blame information for rev 62

Details | Compare with Previous | View Log

Line No. Rev Author Line
1 62 marcus.erl
/******************************************************************************
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
}

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.