3 * ===================================
4 * HARP | Host ATM Research Platform
5 * ===================================
8 * This Host ATM Research Platform ("HARP") file (the "Software") is
9 * made available by Network Computing Services, Inc. ("NetworkCS")
10 * "AS IS". NetworkCS does not provide maintenance, improvements or
11 * support of any kind.
13 * NETWORKCS MAKES NO WARRANTIES OR REPRESENTATIONS, EXPRESS OR IMPLIED,
14 * INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY
15 * AND FITNESS FOR A PARTICULAR PURPOSE, AS TO ANY ELEMENT OF THE
16 * SOFTWARE OR ANY SUPPORT PROVIDED IN CONNECTION WITH THIS SOFTWARE.
17 * In no event shall NetworkCS be responsible for any damages, including
18 * but not limited to consequential damages, arising from or relating to
19 * any use of the Software or related support.
21 * Copyright 1994-1998 Network Computing Services, Inc.
23 * Copies of this Software may be made, however, the above copyright
24 * notice must be reproduced on all copies.
26 * @(#) $FreeBSD: src/sys/netatm/spans/spans_proto.c,v 1.4 1999/08/28 00:48:51 peter Exp $
27 * @(#) $DragonFly: src/sys/netproto/atm/spans/spans_proto.c,v 1.6 2006/01/14 13:36:39 swildner Exp $
31 * SPANS Signalling Manager
32 * ---------------------------
34 * SPANS protocol processing module.
38 #include <netproto/atm/kern_include.h>
40 #include "spans_xdr.h"
41 #include "spans_var.h"
46 caddr_t spans_getname (void *);
47 void spans_connected (void *);
48 void spans_cleared (void *, struct t_atm_cause *);
49 void spans_cpcs_data (void *, KBuffer *);
53 * ATM endpoint for SPANS signalling channel
55 static Atm_endpoint spans_endpt = {
57 ENDPT_SPANS_SIG, /* ep_id */
59 spans_getname, /* ep_getname */
60 spans_connected, /* ep_connected */
61 spans_cleared, /* ep_cleared */
62 NULL, /* ep_incoming */
63 NULL, /* ep_addparty */
64 NULL, /* ep_dropparty */
65 NULL, /* ep_cpcs_ctl */
66 spans_cpcs_data, /* ep_cpcs_data */
67 NULL, /* ep_saal_ctl */
68 NULL, /* ep_saal_data */
69 NULL, /* ep_sscop_ctl */
70 NULL /* ep_sscop_data */
75 * ATM connection attributes for UNI signalling channel
77 static Atm_attributes spans_attr = {
84 T_ATM_PRESENT, /* aal.tag */
85 ATM_AAL3_4 /* aal.aal_type */
88 T_ATM_PRESENT, /* traffic.tag */
90 { /* traffic.v.forward */
91 T_ATM_ABSENT, /* PCR_high */
93 T_ATM_ABSENT, /* SCR_high */
94 T_ATM_ABSENT, /* SCR_all */
95 T_ATM_ABSENT, /* MBS_high */
96 T_ATM_ABSENT, /* MBS_all */
99 { /* traffic.v.backward */
100 T_ATM_ABSENT, /* PCR_high */
102 T_ATM_ABSENT, /* SCR_high */
103 T_ATM_ABSENT, /* SCR_all */
104 T_ATM_ABSENT, /* MBS_high */
105 T_ATM_ABSENT, /* MBS_all */
108 T_YES, /* best_effort */
112 T_ATM_PRESENT, /* bearer.tag */
114 T_ATM_CLASS_X, /* class */
115 T_ATM_NULL, /* traffic_type */
116 T_ATM_NO_END_TO_END, /* timing_req */
118 T_ATM_1_TO_1, /* conn_conf */
122 T_ATM_ABSENT, /* bhli.tag */
125 T_ATM_ABSENT, /* blli.tag_l2 */
126 T_ATM_ABSENT, /* blli.tag_l3 */
129 T_ATM_ABSENT, /* llc.tag */
132 T_ATM_PRESENT, /* called.tag */
135 T_ATM_ABSENT, /* calling.tag */
138 T_ATM_PRESENT, /* qos.tag */
140 T_ATM_NETWORK_CODING, /* coding_standard */
141 { /* qos.v.forward */
142 T_ATM_QOS_CLASS_0, /* class */
144 { /* qos.v.backward */
145 T_ATM_QOS_CLASS_0, /* class */
150 T_ATM_ABSENT, /* transit.tag */
153 T_ATM_ABSENT, /* cause.tag */
159 * SPANS cause structre
161 struct t_atm_cause spans_cause = {
162 T_ATM_ITU_CODING, /* coding_standard */
163 T_ATM_LOC_USER, /* location */
164 T_ATM_CAUSE_UNSPECIFIED_NORMAL, /* cause_value */
165 { 0, 0, 0, 0 } /* diagnostics */
170 * Process a SPANS timeout
172 * Called when a previously scheduled spans control block timer expires.
173 * Processing will based on the current SPANS state.
178 * tip pointer to spans timer control block
185 spans_timer(struct atm_time *tip)
193 * Back-off to SPANS control block
195 spp = (struct spans *)
196 ((caddr_t)tip - (int)(&((struct spans *)0)->sp_time));
198 ATM_DEBUG2("spans_timer: spp=%p,state=%d\n",
202 * Process timeout based on protocol state
204 switch (spp->sp_state) {
209 * Open signalling channel
211 spans_attr.nif = spp->sp_pif->pif_nif;
213 spans_attr.aal.v.aal4.forward_max_SDU_size =
215 spans_attr.aal.v.aal4.backward_max_SDU_size =
217 spans_attr.aal.v.aal4.SSCS_type =
218 T_ATM_SSCS_SSCOP_UNREL;
219 spans_attr.aal.v.aal4.mid_low = 0;
220 spans_attr.aal.v.aal4.mid_high = 0;
222 spans_attr.called.tag = T_ATM_PRESENT;
223 spans_attr.called.addr.address_format = T_ATM_PVC_ADDR;
224 spans_attr.called.addr.address_length =
225 sizeof(Atm_addr_pvc);
226 pvcp = (Atm_addr_pvc *)spans_attr.called.addr.address;
227 ATM_PVC_SET_VPI(pvcp, SPANS_SIG_VPI);
228 ATM_PVC_SET_VCI(pvcp, SPANS_SIG_VCI);
229 spans_attr.called.subaddr.address_format = T_ATM_ABSENT;
230 spans_attr.called.subaddr.address_length = 0;
232 spans_attr.traffic.v.forward.PCR_all_traffic =
233 spp->sp_pif->pif_pcr;
234 spans_attr.traffic.v.backward.PCR_all_traffic =
235 spp->sp_pif->pif_pcr;
237 err = atm_cm_connect(&spans_endpt, spp, &spans_attr,
240 log(LOG_CRIT, "spans: signalling channel setup failed\n");
245 * Signalling channel open, start probing
247 spp->sp_state = SPANS_PROBE;
255 * Send out SPANS_STAT_REQ message
257 msg = (spans_msg *)atm_allocate(&spans_msgpool);
259 /* Retry later if no memory */
260 SPANS_TIMER(spp, SPANS_PROBE_ERR_WAIT);
263 msg->sm_vers = SPANS_VERS_1_0;
264 msg->sm_type = SPANS_STAT_REQ;
265 msg->sm_stat_req.streq_es_epoch = spp->sp_h_epoch;
266 if (spans_send_msg(spp, msg)) {
267 /* Retry later if send fails */
268 SPANS_TIMER(spp, SPANS_PROBE_ERR_WAIT);
276 * Check whether we're getting an answer to our probes
278 if (spp->sp_state == SPANS_ACTIVE &&
279 spp->sp_probe_ct > SPANS_PROBE_THRESH) {
281 * Interface is down, notify VCC owners
283 spans_switch_reset(spp, SPANS_UNI_DOWN);
286 * Set new state and increment host epoch so
287 * switch knows we reset everyting.
289 spp->sp_state = SPANS_PROBE;
295 * Keep sending status requests
297 SPANS_TIMER(spp, SPANS_PROBE_INTERVAL);
303 * Try to terminate the SPANS signalling PVC
305 err = atm_cm_release(spp->sp_conn, &spans_cause);
307 log(LOG_ERR, "spans: can't close signalling channel\n");
312 log(LOG_ERR, "spans: timer state: spp=%p, state=%d\n",
319 * Process a SPANS VCC timeout
321 * Called when a previously scheduled SPANS VCCB timer expires.
322 * Processing will based on the current VCC state.
327 * tip pointer to vccb timer control block
334 spans_vctimer(struct atm_time *tip)
338 struct spans_vccb *svp;
341 * Get VCCB and SPANS control block addresses
343 svp = (struct spans_vccb *) ((caddr_t)tip -
344 (int)(&((struct vccb *)0)->vc_time));
345 spp = (struct spans *)svp->sv_pif->pif_siginst;
347 ATM_DEBUG3("spans_vctimer: svp=%p, sstate=%d, ustate=%d\n",
348 svp, svp->sv_sstate, svp->sv_ustate);
351 * Process timeout based on protocol state
353 switch (svp->sv_sstate) {
357 * Kill the VCCB and notify the owner
359 err = spans_clear_vcc(spp, svp);
366 svp->sv_ustate = VCCU_CLOSED;
367 svp->sv_sstate = SPANS_VC_FREE;
368 spans_free((struct vccb *)svp);
373 * Issued open request, but didn't get response.
375 if (svp->sv_retry < SV_MAX_RETRY) {
377 * Retransmit the open request
379 err = spans_send_open_req(spp, svp);
381 SPANS_VC_TIMER((struct vccb *) svp, SV_TIMEOUT);
384 * Retry limit exceeded--report the open failed
386 svp->sv_ustate = VCCU_CLOSED;
387 svp->sv_sstate = SPANS_VC_FREE;
388 svp->sv_connvc->cvc_attr.cause.tag =
390 svp->sv_connvc->cvc_attr.cause.v.coding_standard =
392 svp->sv_connvc->cvc_attr.cause.v.location =
394 svp->sv_connvc->cvc_attr.cause.v.cause_value =
395 T_ATM_CAUSE_NO_USER_RESPONDING;
396 KM_ZERO(svp->sv_connvc->cvc_attr.cause.v.diagnostics,
397 sizeof(svp->sv_connvc->cvc_attr.cause.v.diagnostics));
398 atm_cm_cleared(svp->sv_connvc);
404 * Issued close request, but didn't get response.
406 if (svp->sv_retry < SV_MAX_RETRY) {
408 * Retransmit the close request
410 err = spans_send_close_req(spp, svp);
412 SPANS_VC_TIMER((struct vccb *) svp, SV_TIMEOUT);
415 * Retry limit exceeded--just finish the close
417 svp->sv_sstate = SPANS_VC_FREE;
418 svp->sv_connvc->cvc_attr.cause.tag = T_ATM_PRESENT;
419 svp->sv_connvc->cvc_attr.cause.v.coding_standard =
421 svp->sv_connvc->cvc_attr.cause.v.location =
423 svp->sv_connvc->cvc_attr.cause.v.cause_value =
424 T_ATM_CAUSE_NO_USER_RESPONDING;
425 KM_ZERO(svp->sv_connvc->cvc_attr.cause.v.diagnostics,
426 sizeof(svp->sv_connvc->cvc_attr.cause.v.diagnostics));
427 atm_cm_cleared(svp->sv_connvc);
431 case SPANS_VC_ACTIVE:
432 case SPANS_VC_ACT_DOWN:
436 log(LOG_ERR, "spans_vctimer: unexpected state %d\n",
441 log(LOG_ERR, "spans: vctimer state: svp=%p, sstate=%d\n",
442 svp, svp->sv_sstate);
451 * tok SPANS signalling channel token (ignored)
454 * pointer to a string identifying the SPANS signalling manager
458 spans_getname(void *tok)
465 * Process a VCC connection notification
467 * Should never be called
470 * tok user's connection token (SPANS protocol block)
477 spans_connected(void *tok)
479 struct spans *spp = (struct spans *)tok;
481 ATM_DEBUG2("spans_connected: spp=%p,state=%d\n",
485 * Connected routine shouldn't ever get called for a PVC
487 log(LOG_ERR, "spans: connected function called, tok=%p\n", spp);
492 * Process a VCC close notification
494 * Called when the SPANS signalling channel is closed
497 * tok user's connection token (spans protocol block)
498 * cp pointer to cause structure
505 spans_cleared(void *tok, struct t_atm_cause *cp)
507 struct spans *spp = (struct spans *)tok;
510 * VCC has been closed.
512 log(LOG_ERR, "spans: signalling channel closed\n");
519 * SPANS CPCS data handler
521 * This is the module which receives data on the SPANS signalling
522 * channel. Processing is based on the indication received from the
523 * AAL and the protocol state.
526 * tok session token (pointer to spans protocol control block)
527 * m pointer to buffer with data
534 spans_cpcs_data(void *tok, KBuffer *m)
536 struct spans *spp = tok;
538 ATM_DEBUG3("spans_cpcs_data: spp=%p,state=%d,m=%p,\n",
539 spp, spp->sp_state, m);
544 spans_rcv_msg(spp, m);