/* * Copyright (c) 1997, 1998 Hellmuth Michaelis. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * *--------------------------------------------------------------------------- * * i4b_l4.c - kernel interface to userland * ----------------------------------------- * * $Id: i4b_l4.c,v 1.33 1998/12/05 18:05:49 hm Exp $ * * last edit-date: [Sat Dec 5 18:35:16 1998] * *---------------------------------------------------------------------------*/ #include "i4b.h" #include "i4bipr.h" #include "i4bisppp.h" #include "i4brbch.h" #include "i4btel.h" #if NI4B > 0 #include #if defined(__FreeBSD__) && __FreeBSD__ >= 3 #include #else #include #endif #include #include #include #include #include #include #include #include #ifdef __FreeBSD__ #include #include #include #else #include #include #include #endif #include #include #include #include #include unsigned int i4b_l4_debug = L4_DEBUG_DEFAULT; struct ctrl_type_desc ctrl_types[CTRL_NUMTYPES] = { { NULL, NULL} }; static int i4b_link_bchandrvr(call_desc_t *cd); static void i4b_unlink_bchandrvr(call_desc_t *cd); static void i4b_l4_setup_timeout(call_desc_t *cd); /*---------------------------------------------------------------------------* * send MSG_PDEACT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_pdeact(int controller, int numactive) { struct mbuf *m; int i; call_desc_t *cd; for(i=0; i < N_CALL_DESC; i++) { if((call_desc[i].cdid != CDID_UNUSED) && (ctrl_desc[call_desc[i].controller].ctrl_type == CTRL_PASSIVE) && (ctrl_desc[call_desc[i].controller].unit == controller)) { cd = &call_desc[i]; if(cd->timeout_active) { #if defined(__FreeBSD__) && __FreeBSD__ >= 3 untimeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, cd->idle_timeout_handle); #else untimeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd); #endif } if(cd->dlt != NULL) { (*cd->dlt->line_disconnected)(cd->driver_unit, (void *)cd); i4b_unlink_bchandrvr(cd); } if((cd->channelid == CHAN_B1) || (cd->channelid == CHAN_B2)) { ctrl_desc[cd->controller].bch_state[cd->channelid] = BCH_ST_FREE; } cd->cdid = CDID_UNUSED; } } if((m = i4b_Dgetmbuf(sizeof(msg_pdeact_ind_t))) != NULL) { msg_pdeact_ind_t *md = (msg_pdeact_ind_t *)m->m_data; md->header.type = MSG_PDEACT_IND; md->header.cdid = -1; md->controller = controller; md->numactive = numactive; i4bputqueue_hipri(m); /* URGENT !!! */ } } /*---------------------------------------------------------------------------* * send MSG_L12STAT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_l12stat(int controller, int layer, int state) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_l12stat_ind_t))) != NULL) { msg_l12stat_ind_t *md = (msg_l12stat_ind_t *)m->m_data; md->header.type = MSG_L12STAT_IND; md->header.cdid = -1; md->controller = controller; md->layer = layer; md->state = state; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_TEIASG_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_teiasg(int controller, int tei) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_teiasg_ind_t))) != NULL) { msg_teiasg_ind_t *md = (msg_teiasg_ind_t *)m->m_data; md->header.type = MSG_TEIASG_IND; md->header.cdid = -1; md->controller = controller; md->tei = ctrl_desc[controller].tei; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_DIALOUT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_dialout(int driver, int driver_unit) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_dialout_ind_t))) != NULL) { msg_dialout_ind_t *md = (msg_dialout_ind_t *)m->m_data; md->header.type = MSG_DIALOUT_IND; md->header.cdid = -1; md->driver = driver; md->driver_unit = driver_unit; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_NEGOTIATION_COMPL message to userland *---------------------------------------------------------------------------*/ void i4b_l4_negcomplete(call_desc_t *cd) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_negcomplete_ind_t))) != NULL) { msg_negcomplete_ind_t *md = (msg_negcomplete_ind_t *)m->m_data; md->header.type = MSG_NEGCOMP_IND; md->header.cdid = cd->cdid; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_IFSTATE_CHANGED_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_ifstate_changed(call_desc_t *cd, int new_state) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_ifstatechg_ind_t))) != NULL) { msg_ifstatechg_ind_t *md = (msg_ifstatechg_ind_t *)m->m_data; md->header.type = MSG_IFSTATE_CHANGED_IND; md->header.cdid = cd->cdid; md->state = new_state; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_DRVRDISC_REQ message to userland *---------------------------------------------------------------------------*/ void i4b_l4_drvrdisc(int driver, int driver_unit) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_drvrdisc_req_t))) != NULL) { msg_drvrdisc_req_t *md = (msg_drvrdisc_req_t *)m->m_data; md->header.type = MSG_DRVRDISC_REQ; md->header.cdid = -1; md->driver = driver; md->driver_unit = driver_unit; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_ACCT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_accounting(int driver, int driver_unit, int accttype, int ioutbytes, int iinbytes, int ro, int ri, int outbytes, int inbytes) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_accounting_ind_t))) != NULL) { msg_accounting_ind_t *md = (msg_accounting_ind_t *)m->m_data; md->header.type = MSG_ACCT_IND; md->header.cdid = -1; md->driver = driver; md->driver_unit = driver_unit; md->accttype = accttype; md->ioutbytes = ioutbytes; md->iinbytes = iinbytes; md->outbps = ro; md->inbps = ri; md->outbytes = outbytes; md->inbytes = inbytes; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_CONNECT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_connect_ind(call_desc_t *cd) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_connect_ind_t))) != NULL) { msg_connect_ind_t *mp = (msg_connect_ind_t *)m->m_data; mp->header.type = MSG_CONNECT_IND; mp->header.cdid = cd->cdid; mp->controller = cd->controller; mp->channel = cd->channelid; mp->bprot = cd->bprot; cd->dir = DIR_INCOMING; if(strlen(cd->dst_telno) > 0) strcpy(mp->dst_telno, cd->dst_telno); else strcpy(mp->dst_telno, TELNO_EMPTY); if(strlen(cd->src_telno) > 0) strcpy(mp->src_telno, cd->src_telno); else strcpy(mp->src_telno, TELNO_EMPTY); strcpy(mp->display, cd->display); mp->scr_ind = cd->scr_ind; T400_start(cd); i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_CONNECT_ACTIVE_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_connect_active_ind(call_desc_t *cd) { int s; struct mbuf *m; s = SPLI4B(); cd->last_active_time = cd->connect_time = SECOND; DBGL4(L4_TIMO, "i4b_l4_connect_active_ind", ("last_active/connect_time=%ld\n", (long)cd->connect_time)); i4b_link_bchandrvr(cd); (*cd->dlt->line_connected)(cd->driver_unit, (void *)cd); i4b_l4_setup_timeout(cd); splx(s); if((m = i4b_Dgetmbuf(sizeof(msg_connect_active_ind_t))) != NULL) { msg_connect_active_ind_t *mp = (msg_connect_active_ind_t *)m->m_data; mp->header.type = MSG_CONNECT_ACTIVE_IND; mp->header.cdid = cd->cdid; mp->controller = cd->controller; mp->channel = cd->channelid; if(cd->datetime[0] != '\0') strcpy(mp->datetime, cd->datetime); else mp->datetime[0] = '\0'; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_DISCONNECT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_disconnect_ind(call_desc_t *cd) { struct mbuf *m; if(cd->timeout_active) #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 untimeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, cd->idle_timeout_handle); #else untimeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd); #endif if(cd->dlt != NULL) { (*cd->dlt->line_disconnected)(cd->driver_unit, (void *)cd); i4b_unlink_bchandrvr(cd); } if((cd->channelid == CHAN_B1) || (cd->channelid == CHAN_B2)) { ctrl_desc[cd->controller].bch_state[cd->channelid] = BCH_ST_FREE; } else { /* no error, might be hunting call for callback */ DBGL4(L4_MSG, "i4b_l4_disconnect_ind", ("channel free not B1/B2 but %d!\n", cd->channelid)); } if((m = i4b_Dgetmbuf(sizeof(msg_disconnect_ind_t))) != NULL) { msg_disconnect_ind_t *mp = (msg_disconnect_ind_t *)m->m_data; mp->header.type = MSG_DISCONNECT_IND; mp->header.cdid = cd->cdid; mp->cause = cd->cause_in; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_IDLE_TIMEOUT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_idle_timeout_ind(call_desc_t *cd) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_idle_timeout_ind_t))) != NULL) { msg_idle_timeout_ind_t *mp = (msg_idle_timeout_ind_t *)m->m_data; mp->header.type = MSG_IDLE_TIMEOUT_IND; mp->header.cdid = cd->cdid; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_CHARGING_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_charging_ind(call_desc_t *cd) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_charging_ind_t))) != NULL) { msg_charging_ind_t *mp = (msg_charging_ind_t *)m->m_data; mp->header.type = MSG_CHARGING_IND; mp->header.cdid = cd->cdid; mp->units_type = cd->units_type; /*XXX*/ if(mp->units_type == CHARGE_CALC) mp->units = cd->cunits; else mp->units = cd->units; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_STATUS_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_status_ind(call_desc_t *cd) { } /*---------------------------------------------------------------------------* * send MSG_ALERT_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_alert_ind(call_desc_t *cd) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_alert_ind_t))) != NULL) { msg_alert_ind_t *mp = (msg_alert_ind_t *)m->m_data; mp->header.type = MSG_ALERT_IND; mp->header.cdid = cd->cdid; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * send MSG_INFO_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_info_ind(call_desc_t *cd) { } /*---------------------------------------------------------------------------* * send MSG_INFO_IND message to userland *---------------------------------------------------------------------------*/ void i4b_l4_proceeding_ind(call_desc_t *cd) { struct mbuf *m; if((m = i4b_Dgetmbuf(sizeof(msg_proceeding_ind_t))) != NULL) { msg_proceeding_ind_t *mp = (msg_proceeding_ind_t *)m->m_data; mp->header.type = MSG_PROCEEDING_IND; mp->header.cdid = cd->cdid; mp->controller = cd->controller; mp->channel = cd->channelid; i4bputqueue(m); } } /*---------------------------------------------------------------------------* * link a driver(unit) to a B-channel(controller,unit,channel) *---------------------------------------------------------------------------*/ static int i4b_link_bchandrvr(call_desc_t *cd) { int t = ctrl_desc[cd->controller].ctrl_type; if(t < 0 || t >= CTRL_NUMTYPES || ctrl_types[t].get_linktab == NULL) { cd->ilt = NULL; } else { cd->ilt = ctrl_types[t].get_linktab( ctrl_desc[cd->controller].unit, cd->channelid); } switch(cd->driver) { #if NI4BRBCH > 0 case BDRV_RBCH: cd->dlt = rbch_ret_linktab(cd->driver_unit); break; #endif #if NI4BTEL > 0 case BDRV_TEL: cd->dlt = tel_ret_linktab(cd->driver_unit); break; #endif #if NI4BIPR > 0 case BDRV_IPR: cd->dlt = ipr_ret_linktab(cd->driver_unit); break; #endif #if NI4BISPPP > 0 case BDRV_ISPPP: cd->dlt = i4bisppp_ret_linktab(cd->driver_unit); break; #endif default: cd->dlt = NULL; break; } if(cd->dlt == NULL || cd->ilt == NULL) return(-1); if(t >= 0 && t < CTRL_NUMTYPES && ctrl_types[t].set_linktab != NULL) { ctrl_types[t].set_linktab( ctrl_desc[cd->controller].unit, cd->channelid, cd->dlt); } switch(cd->driver) { #if NI4BRBCH > 0 case BDRV_RBCH: rbch_set_linktab(cd->driver_unit, cd->ilt); break; #endif #if NI4BTEL > 0 case BDRV_TEL: tel_set_linktab(cd->driver_unit, cd->ilt); break; #endif #if NI4BIPR > 0 case BDRV_IPR: ipr_set_linktab(cd->driver_unit, cd->ilt); break; #endif #if NI4BISPPP > 0 case BDRV_ISPPP: i4bisppp_set_linktab(cd->driver_unit, cd->ilt); break; #endif default: return(0); break; } /* activate B channel */ (*cd->ilt->bch_config)(ctrl_desc[cd->controller].unit, cd->channelid, cd->bprot, 1); return(0); } /*---------------------------------------------------------------------------* * unlink a driver(unit) from a B-channel(controller,unit,channel) *---------------------------------------------------------------------------*/ static void i4b_unlink_bchandrvr(call_desc_t *cd) { int t = ctrl_desc[cd->controller].ctrl_type; if(t < 0 || t >= CTRL_NUMTYPES || ctrl_types[t].get_linktab == NULL) { cd->ilt = NULL; return; } else { cd->ilt = ctrl_types[t].get_linktab( ctrl_desc[cd->controller].unit, cd->channelid); } /* deactivate B channel */ (*cd->ilt->bch_config)(ctrl_desc[cd->controller].unit, cd->channelid, cd->bprot, 0); } /*--------------------------------------------------------------------------- How shorthold mode works for OUTGOING connections ================================================= |<---- unchecked-window ------->|<-checkwindow->|<-safetywindow>| idletime_state: IST_NONCHK IST_CHECK IST_SAFE | | | | time>>+-------------------------------+---------------+---------------+-... | | | | | |<--idle_time-->|<--earlyhup--->| |<-----------------------unitlen------------------------------->| unitlen - specifies the time a charging unit lasts idle_time - specifies the thime the line must be idle at the end of the unit to be elected for hangup earlyhup - is the beginning of a timing safety zone before the next charging unit starts The algorithm works as follows: lets assume the unitlen is 100 secons, idle_time is 40 seconds and earlyhup is 10 seconds. The line then must be idle 50 seconds after the begin of the current unit and it must then be quiet for 40 seconds. if it has been quiet for this 40 seconds, the line is closed 10 seconds before the next charging unit starts. In case there was any traffic within the idle_time, the line is not closed. It does not matter whether there was any traffic between second 0 and second 50 or not. How shorthold mode works for INCOMING connections ================================================= it is just possible to specify a maximum idle time for incoming connections, after this time of no activity on the line the line is closed. ---------------------------------------------------------------------------*/ /*---------------------------------------------------------------------------* * B channel idle check timeout setup *---------------------------------------------------------------------------*/ static void i4b_l4_setup_timeout(call_desc_t *cd) { cd->timeout_active = 0; cd->idletime_state = IST_IDLE; if((cd->dir == DIR_INCOMING) && (cd->max_idle_time > 0)) { /* incoming call: simple max idletime check */ #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz/2); cd->timeout_active = 1; DBGL4(L4_TIMO, "i4b_l4_setup_timeout", ("%ld: incoming-call, setup max_idle_time to %ld\n", (long)SECOND, (long)cd->max_idle_time)); } else if((cd->dir == DIR_OUTGOING) && (cd->idle_time > 0)) { /* outgoing call */ if((cd->idle_time > 0) && (cd->unitlen_time == 0)) { /* outgoing call: simple max idletime check */ #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz/2); cd->timeout_active = 1; DBGL4(L4_TIMO, "i4b_l4_setup_timeout", ("%ld: outgoing-call, setup idle_time to %ld\n", (long)SECOND, (long)cd->idle_time)); } else if((cd->unitlen_time > 0) && (cd->unitlen_time > (cd->idle_time + cd->earlyhup_time))) { /* outgoing call: full shorthold mode check */ #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz*(cd->unitlen_time - (cd->idle_time + cd->earlyhup_time))); cd->timeout_active = 1; cd->idletime_state = IST_NONCHK; DBGL4(L4_TIMO, "i4b_l4_setup_timeout", ("%ld: outgoing-call, start %ld sec nocheck window\n", (long)SECOND, (long)(cd->unitlen_time - (cd->idle_time + cd->earlyhup_time)))); if(cd->aocd_flag == 0) { cd->units_type = CHARGE_CALC; cd->cunits++; i4b_l4_charging_ind(cd); } } else { /* parms somehow got wrong .. */ DBGL4(L4_ERR, "i4b_l4_setup_timeout", ("%ld: ERROR: idletime[%ld]+earlyhup[%ld] > unitlength[%ld]!\n", (long)SECOND, (long)cd->idle_time, (long)cd->earlyhup_time, (long)cd->unitlen_time)); } } else { DBGL4(L4_TIMO, "i4b_l4_setup_timeout", ("no idle_timeout configured\n")); } } /*---------------------------------------------------------------------------* * B channel idle check timeout function *---------------------------------------------------------------------------*/ void i4b_idle_check(call_desc_t *cd) { int s; if(cd->cdid == CDID_UNUSED) return; s = SPLI4B(); /* failsafe */ if(cd->timeout_active == 0) { DBGL4(L4_ERR, "i4b_idle_check", ("ERROR: timeout_active == 0 !!!\n")); } else { cd->timeout_active = 0; } /* incoming connections, simple idletime check */ if(cd->dir == DIR_INCOMING) { if((cd->last_active_time + cd->max_idle_time) <= SECOND) { DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: incoming-call, line idle timeout, disconnecting!\n", (long)SECOND)); (*ctrl_desc[cd->controller].N_DISCONNECT_REQUEST)(cd->cdid, (CAUSET_I4B << 8) | CAUSE_I4B_NORMAL); i4b_l4_idle_timeout_ind(cd); } else { DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: incoming-call, activity, last_active=%ld, max_idle=%ld\n", (long)SECOND, (long)cd->last_active_time, (long)cd->max_idle_time)); #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz/2); cd->timeout_active = 1; } } /* outgoing connections */ else if(cd->dir == DIR_OUTGOING) { /* simple idletime calculation */ if((cd->idle_time > 0) && (cd->unitlen_time == 0)) { if((cd->last_active_time + cd->idle_time) <= SECOND) { DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: outgoing-call-st, idle timeout, disconnecting!\n", (long)SECOND)); (*ctrl_desc[cd->controller].N_DISCONNECT_REQUEST)(cd->cdid, (CAUSET_I4B << 8) | CAUSE_I4B_NORMAL); i4b_l4_idle_timeout_ind(cd); } else { DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: outgoing-call-st, activity, last_active=%ld, max_idle=%ld\n", (long)SECOND, (long)cd->last_active_time, (long)cd->idle_time)); #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz/2); cd->timeout_active = 1; } } /* full shorthold mode calculation */ else if((cd->unitlen_time > 0) && (cd->unitlen_time > (cd->idle_time + cd->earlyhup_time))) { switch(cd->idletime_state) { case IST_NONCHK: /* end of non-check time */ #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz*(cd->idle_time)); cd->idletimechk_start = SECOND; cd->idletime_state = IST_CHECK; cd->timeout_active = 1; DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: outgoing-call, idletime check window reached!\n", (long)SECOND)); break; case IST_CHECK: /* end of idletime chk */ if((cd->last_active_time > cd->idletimechk_start) && (cd->last_active_time <= SECOND)) { /* activity detected */ #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz*(cd->earlyhup_time)); cd->timeout_active = 1; cd->idletime_state = IST_SAFE; DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: outgoing-call, activity at %ld, wait earlyhup-end\n", (long)SECOND, (long)cd->last_active_time)); } else { /* no activity, hangup */ DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: outgoing-call, idle timeout, last activity at %ld\n", (long)SECOND, (long)cd->last_active_time)); (*ctrl_desc[cd->controller].N_DISCONNECT_REQUEST)(cd->cdid, (CAUSET_I4B << 8) | CAUSE_I4B_NORMAL); i4b_l4_idle_timeout_ind(cd); cd->idletime_state = IST_IDLE; } break; case IST_SAFE: /* end of earlyhup time */ #if defined(__FreeBSD_version) && __FreeBSD_version >= 300001 cd->idle_timeout_handle = #endif timeout((TIMEOUT_FUNC_T)i4b_idle_check,(void *)cd, hz*(cd->unitlen_time - (cd->idle_time+cd->earlyhup_time))); cd->timeout_active = 1; cd->idletime_state = IST_NONCHK; if(cd->aocd_flag == 0) { cd->units_type = CHARGE_CALC; cd->cunits++; i4b_l4_charging_ind(cd); } DBGL4(L4_TIMO, "i4b_idle_check", ("%ld: outgoing-call, earlyhup end, wait for idletime start\n", (long)SECOND)); break; default: DBGL4(L4_ERR, "i4b_idle_check", ("outgoing-call: invalid idletime_state value!\n")); cd->idletime_state = IST_IDLE; break; } } } splx(s); } #endif /* NI4B > 0 */