2001-01-05 15:49:52 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1997, 2000 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
|
|
|
|
* -----------------------------------------
|
|
|
|
*
|
2004-03-21 19:29:40 +03:00
|
|
|
* $Id: i4b_l4.c,v 1.26 2004/03/21 16:29:40 martin Exp $
|
2001-01-05 15:49:52 +03:00
|
|
|
*
|
|
|
|
* $FreeBSD$
|
|
|
|
*
|
|
|
|
* last edit-date: [Fri Jan 5 11:33:47 2001]
|
|
|
|
*
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
|
2001-11-13 03:56:55 +03:00
|
|
|
#include <sys/cdefs.h>
|
2004-03-21 19:29:40 +03:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: i4b_l4.c,v 1.26 2004/03/21 16:29:40 martin Exp $");
|
2001-11-13 03:56:55 +03:00
|
|
|
|
2002-03-16 19:55:51 +03:00
|
|
|
#include "isdn.h"
|
|
|
|
#include "irip.h"
|
2001-01-05 15:49:52 +03:00
|
|
|
|
2002-03-16 19:55:51 +03:00
|
|
|
#if NISDN > 0
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/conf.h>
|
|
|
|
#include <sys/mbuf.h>
|
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/fcntl.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <net/if.h>
|
|
|
|
|
|
|
|
#if defined(__NetBSD__) && __NetBSD_Version__ >= 104230000
|
|
|
|
#include <sys/callout.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __FreeBSD__
|
|
|
|
#include <machine/i4b_debug.h>
|
|
|
|
#include <machine/i4b_ioctl.h>
|
|
|
|
#include <machine/i4b_cause.h>
|
|
|
|
#else
|
|
|
|
#include <netisdn/i4b_debug.h>
|
|
|
|
#include <netisdn/i4b_ioctl.h>
|
|
|
|
#include <netisdn/i4b_cause.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <netisdn/i4b_global.h>
|
|
|
|
#include <netisdn/i4b_l3l4.h>
|
|
|
|
#include <netisdn/i4b_mbuf.h>
|
2002-02-14 19:46:23 +03:00
|
|
|
#include <netisdn/i4b_l2.h>
|
2001-01-05 15:49:52 +03:00
|
|
|
#include <netisdn/i4b_l3.h>
|
|
|
|
#include <netisdn/i4b_l4.h>
|
|
|
|
|
|
|
|
unsigned int i4b_l4_debug = L4_DEBUG_DEFAULT;
|
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
/*
|
2003-10-03 20:38:44 +04:00
|
|
|
* ISDNs, in userland sometimes called "controllers", but one controller
|
|
|
|
* may have multiple BRIs, for example daic QUAD cards attach four BRIs.
|
|
|
|
* An ISDN may also be a PRI (30 B channels).
|
2002-03-24 23:35:43 +03:00
|
|
|
*/
|
2003-10-03 20:38:44 +04:00
|
|
|
static SLIST_HEAD(, isdn_l3_driver) isdnif_list = SLIST_HEAD_INITIALIZER(isdnif_list);
|
|
|
|
static int next_isdnif = 0;
|
2002-03-24 23:35:43 +03:00
|
|
|
|
|
|
|
/*
|
2003-10-03 20:38:44 +04:00
|
|
|
* Attach a new L3 driver instance and return its ISDN identifier
|
2002-03-24 23:35:43 +03:00
|
|
|
*/
|
|
|
|
struct isdn_l3_driver *
|
2003-10-03 20:38:44 +04:00
|
|
|
isdn_attach_isdnif(const char *devname, const char *cardname,
|
|
|
|
void *l1_token, const struct isdn_l3_driver_functions *l3driver, int nbch)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
2002-03-25 15:07:33 +03:00
|
|
|
int s = splnet();
|
2003-10-03 20:38:44 +04:00
|
|
|
int i, l, isdnif = next_isdnif++;
|
2002-03-24 23:35:43 +03:00
|
|
|
struct isdn_l3_driver *new_ctrl;
|
|
|
|
|
|
|
|
new_ctrl = malloc(sizeof(*new_ctrl), M_DEVBUF, 0);
|
|
|
|
memset(new_ctrl, 0, sizeof *new_ctrl);
|
2003-10-03 20:38:44 +04:00
|
|
|
SLIST_INSERT_HEAD(&isdnif_list, new_ctrl, l3drvq);
|
2002-03-24 23:35:43 +03:00
|
|
|
l = strlen(devname);
|
2003-05-16 09:12:32 +04:00
|
|
|
new_ctrl->devname = malloc(l + 1, M_DEVBUF, 0);
|
|
|
|
strlcpy(new_ctrl->devname, devname, l + 1);
|
2002-03-24 23:35:43 +03:00
|
|
|
l = strlen(cardname);
|
2003-05-16 09:12:32 +04:00
|
|
|
new_ctrl->card_name = malloc(l + 1, M_DEVBUF, 0);
|
|
|
|
strlcpy(new_ctrl->card_name, cardname, l + 1);
|
2002-03-24 23:35:43 +03:00
|
|
|
|
|
|
|
new_ctrl->l3driver = l3driver;
|
|
|
|
new_ctrl->l1_token = l1_token;
|
2003-10-03 20:38:44 +04:00
|
|
|
new_ctrl->isdnif = isdnif;
|
2002-03-25 23:28:41 +03:00
|
|
|
new_ctrl->tei = -1;
|
2002-03-24 23:35:43 +03:00
|
|
|
new_ctrl->dl_est = DL_DOWN;
|
2003-10-03 20:38:44 +04:00
|
|
|
new_ctrl->nbch = nbch;
|
|
|
|
|
|
|
|
new_ctrl->bch_state = malloc(nbch * sizeof(int), M_DEVBUF, 0);
|
|
|
|
for (i = 0; i < nbch; i++)
|
|
|
|
new_ctrl->bch_state[i] = BCH_ST_FREE;
|
2002-03-24 23:35:43 +03:00
|
|
|
|
2002-03-25 15:07:33 +03:00
|
|
|
splx(s);
|
2002-03-24 23:35:43 +03:00
|
|
|
|
|
|
|
return new_ctrl;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Detach a L3 driver instance
|
|
|
|
*/
|
|
|
|
int
|
2003-10-03 20:38:44 +04:00
|
|
|
isdn_detach_isdnif(struct isdn_l3_driver *l3drv)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
2002-03-25 15:07:33 +03:00
|
|
|
struct isdn_l3_driver *sc;
|
|
|
|
int s = splnet();
|
2003-10-03 20:38:44 +04:00
|
|
|
int isdnif = l3drv->isdnif;
|
2002-03-25 15:07:33 +03:00
|
|
|
int max;
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
i4b_l4_contr_ev_ind(isdnif, 0);
|
|
|
|
SLIST_REMOVE(&isdnif_list, l3drv, isdn_l3_driver, l3drvq);
|
2002-03-25 15:07:33 +03:00
|
|
|
|
|
|
|
max = -1;
|
2003-10-03 20:38:44 +04:00
|
|
|
SLIST_FOREACH(sc, &isdnif_list, l3drvq)
|
|
|
|
if (sc->isdnif > max)
|
|
|
|
max = sc->isdnif;
|
|
|
|
next_isdnif = max+1;
|
2002-03-25 15:07:33 +03:00
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
free_all_cd_of_isdnif(isdnif);
|
2002-03-27 10:39:35 +03:00
|
|
|
|
2002-03-25 15:07:33 +03:00
|
|
|
splx(s);
|
|
|
|
|
2002-03-25 17:25:06 +03:00
|
|
|
free(l3drv, M_DEVBUF);
|
2003-10-03 20:38:44 +04:00
|
|
|
printf("ISDN %d detached\n", isdnif);
|
2002-03-24 23:35:43 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct isdn_l3_driver *
|
2003-10-03 20:38:44 +04:00
|
|
|
isdn_find_l3_by_isdnif(int isdnif)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
|
|
|
struct isdn_l3_driver *sc;
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
SLIST_FOREACH(sc, &isdnif_list, l3drvq)
|
|
|
|
if (sc->isdnif == isdnif)
|
2002-03-24 23:35:43 +03:00
|
|
|
return sc;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
int isdn_count_isdnif(int *misdnif)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
|
|
|
struct isdn_l3_driver *sc;
|
|
|
|
int count = 0;
|
2003-10-03 20:38:44 +04:00
|
|
|
int max_isdnif = -1;
|
2002-03-24 23:35:43 +03:00
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
SLIST_FOREACH(sc, &isdnif_list, l3drvq) {
|
2002-03-24 23:35:43 +03:00
|
|
|
count++;
|
2003-10-03 20:38:44 +04:00
|
|
|
if (sc->isdnif > max_isdnif)
|
|
|
|
max_isdnif = sc->isdnif;
|
2002-03-24 23:35:43 +03:00
|
|
|
}
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
if (misdnif)
|
|
|
|
*misdnif = max_isdnif;
|
2002-03-24 23:35:43 +03:00
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *
|
2003-10-03 20:38:44 +04:00
|
|
|
isdn_find_softc_by_isdnif(int isdnif)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
2003-10-03 20:38:44 +04:00
|
|
|
struct isdn_l3_driver *sc = isdn_find_l3_by_isdnif(isdnif);
|
2002-03-24 23:35:43 +03:00
|
|
|
if (sc == NULL)
|
|
|
|
return NULL;
|
|
|
|
/*
|
|
|
|
* XXX - hack: do not return a softc for active cards.
|
|
|
|
* all callers of this expecting l2_softc* results
|
|
|
|
* should be fixed!
|
|
|
|
*/
|
|
|
|
if (sc->l3driver->N_DOWNLOAD)
|
|
|
|
return NULL;
|
|
|
|
return sc->l1_token;
|
|
|
|
}
|
2001-01-05 15:49:52 +03:00
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* daemon is attached
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
i4b_l4_daemon_attached(void)
|
|
|
|
{
|
|
|
|
struct isdn_l3_driver *d;
|
|
|
|
|
|
|
|
int x = splnet();
|
2003-10-03 20:38:44 +04:00
|
|
|
SLIST_FOREACH(d, &isdnif_list, l3drvq)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
2002-03-30 14:15:41 +03:00
|
|
|
d->l3driver->N_MGMT_COMMAND(d, CMR_DOPEN, 0);
|
2002-03-24 23:35:43 +03:00
|
|
|
}
|
|
|
|
splx(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* daemon is detached
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
i4b_l4_daemon_detached(void)
|
|
|
|
{
|
|
|
|
struct isdn_l3_driver *d;
|
|
|
|
|
|
|
|
int x = splnet();
|
2003-10-03 20:38:44 +04:00
|
|
|
SLIST_FOREACH(d, &isdnif_list, l3drvq)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
2002-03-30 14:15:41 +03:00
|
|
|
d->l3driver->N_MGMT_COMMAND(d, CMR_DCLOSE, 0);
|
2002-03-24 23:35:43 +03:00
|
|
|
}
|
|
|
|
splx(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* B-channel layer 4 drivers and their registry.
|
|
|
|
* (Application drivers connecting to a B-channel)
|
|
|
|
*/
|
2001-01-05 15:49:52 +03:00
|
|
|
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);
|
|
|
|
static void i4b_idle_check_fix_unit(call_desc_t *cd);
|
|
|
|
static void i4b_idle_check_var_unit(call_desc_t *cd);
|
|
|
|
static void i4b_l4_setup_timeout_fix_unit(call_desc_t *cd);
|
|
|
|
static void i4b_l4_setup_timeout_var_unit(call_desc_t *cd);
|
|
|
|
static time_t i4b_get_idletime(call_desc_t *cd);
|
|
|
|
|
2002-03-17 23:54:04 +03:00
|
|
|
static int next_l4_driver_id = 0;
|
|
|
|
|
|
|
|
struct l4_driver_desc {
|
2002-03-25 15:07:33 +03:00
|
|
|
SLIST_ENTRY(l4_driver_desc) l4drvq;
|
2002-03-17 23:54:04 +03:00
|
|
|
char name[L4DRIVER_NAME_SIZ];
|
|
|
|
int driver_id;
|
|
|
|
const struct isdn_l4_driver_functions *driver;
|
|
|
|
int units;
|
|
|
|
};
|
2002-03-25 15:07:33 +03:00
|
|
|
static SLIST_HEAD(, l4_driver_desc) l4_driver_registry
|
|
|
|
= SLIST_HEAD_INITIALIZER(l4_driver_registry);
|
2002-03-17 23:54:04 +03:00
|
|
|
|
|
|
|
int isdn_l4_driver_attach(const char *name, int units, const struct isdn_l4_driver_functions *driver)
|
|
|
|
{
|
|
|
|
struct l4_driver_desc * new_driver;
|
|
|
|
|
|
|
|
new_driver = malloc(sizeof(struct l4_driver_desc), M_DEVBUF, 0);
|
|
|
|
memset(new_driver, 0, sizeof(struct l4_driver_desc));
|
|
|
|
strncpy(new_driver->name, name, L4DRIVER_NAME_SIZ);
|
|
|
|
new_driver->name[L4DRIVER_NAME_SIZ-1] = 0;
|
|
|
|
new_driver->driver_id = next_l4_driver_id++;
|
|
|
|
new_driver->driver = driver;
|
|
|
|
new_driver->units = units;
|
2002-03-25 15:07:33 +03:00
|
|
|
SLIST_INSERT_HEAD(&l4_driver_registry, new_driver, l4drvq);
|
2002-03-17 23:54:04 +03:00
|
|
|
return new_driver->driver_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
int isdn_l4_driver_detatch(const char *name)
|
|
|
|
{
|
|
|
|
/* XXX - not yet implemented */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct isdn_l4_driver_functions *isdn_l4_find_driver(const char *name, int unit)
|
|
|
|
{
|
|
|
|
struct l4_driver_desc * d;
|
2002-03-25 15:07:33 +03:00
|
|
|
SLIST_FOREACH(d, &l4_driver_registry, l4drvq)
|
2002-03-17 23:54:04 +03:00
|
|
|
if (strcmp(d->name, name) == 0) {
|
|
|
|
return d->driver;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int isdn_l4_find_driverid(const char *name)
|
|
|
|
{
|
|
|
|
struct l4_driver_desc * d;
|
2002-03-25 15:07:33 +03:00
|
|
|
SLIST_FOREACH(d, &l4_driver_registry, l4drvq)
|
2002-03-17 23:54:04 +03:00
|
|
|
if (strcmp(d->name, name) == 0) {
|
|
|
|
return d->driver_id;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct isdn_l4_driver_functions *isdn_l4_get_driver(int driver_id, int unit)
|
|
|
|
{
|
|
|
|
struct l4_driver_desc * d;
|
2002-03-25 15:07:33 +03:00
|
|
|
SLIST_FOREACH(d, &l4_driver_registry, l4drvq)
|
2002-03-17 23:54:04 +03:00
|
|
|
if (d->driver_id == driver_id) {
|
|
|
|
return d->driver;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2001-01-05 15:49:52 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* send MSG_PDEACT_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
2002-05-21 14:31:10 +04:00
|
|
|
i4b_l4_pdeact(struct isdn_l3_driver *d, int numactive)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
struct mbuf *m;
|
|
|
|
int i;
|
|
|
|
call_desc_t *cd;
|
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
for(i=0; i < num_call_desc; i++)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-05-21 14:31:10 +04:00
|
|
|
if(call_desc[i].cdid != CDID_UNUSED && call_desc[i].l3drv == d)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
cd = &call_desc[i];
|
|
|
|
|
|
|
|
if(cd->timeout_active)
|
|
|
|
{
|
|
|
|
STOP_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd);
|
|
|
|
}
|
|
|
|
|
2002-03-17 23:54:04 +03:00
|
|
|
if (cd->l4_driver != NULL && cd->l4_driver_softc != NULL)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-17 23:54:04 +03:00
|
|
|
(*cd->l4_driver->line_disconnected)(cd->l4_driver_softc, (void *)cd);
|
2001-01-05 15:49:52 +03:00
|
|
|
i4b_unlink_bchandrvr(cd);
|
|
|
|
}
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
if ((cd->channelid >= 0)
|
|
|
|
&& (cd->channelid < d->nbch));
|
2002-03-24 23:35:43 +03:00
|
|
|
d->bch_state[cd->channelid] = BCH_ST_FREE;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
md->controller = d->isdnif;
|
2001-01-05 15:49:52 +03:00
|
|
|
md->numactive = numactive;
|
|
|
|
|
|
|
|
i4bputqueue_hipri(m); /* URGENT !!! */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* send MSG_L12STAT_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
2002-05-21 14:31:10 +04:00
|
|
|
i4b_l4_l12stat(struct isdn_l3_driver *d, int layer, int state)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
md->controller = d->isdnif;
|
2001-01-05 15:49:52 +03:00
|
|
|
md->layer = layer;
|
|
|
|
md->state = state;
|
|
|
|
|
|
|
|
i4bputqueue(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* send MSG_TEIASG_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
2002-05-21 14:31:10 +04:00
|
|
|
i4b_l4_teiasg(struct isdn_l3_driver *d, int tei)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
md->controller = d->isdnif;
|
2002-03-24 23:35:43 +03:00
|
|
|
md->tei = d->tei;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
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_DIALOUTNUMBER_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
i4b_l4_dialoutnumber(int driver, int driver_unit, int cmdlen, char *cmd)
|
|
|
|
{
|
|
|
|
struct mbuf *m;
|
|
|
|
|
|
|
|
if((m = i4b_Dgetmbuf(sizeof(msg_dialoutnumber_ind_t))) != NULL)
|
|
|
|
{
|
|
|
|
msg_dialoutnumber_ind_t *md = (msg_dialoutnumber_ind_t *)m->m_data;
|
|
|
|
|
|
|
|
md->header.type = MSG_DIALOUTNUMBER_IND;
|
|
|
|
md->header.cdid = -1;
|
|
|
|
|
|
|
|
md->driver = driver;
|
|
|
|
md->driver_unit = driver_unit;
|
|
|
|
|
|
|
|
if(cmdlen > TELNO_MAX)
|
|
|
|
cmdlen = TELNO_MAX;
|
|
|
|
|
|
|
|
md->cmdlen = cmdlen;
|
2001-07-08 14:33:58 +04:00
|
|
|
memcpy(md->cmd, cmd, cmdlen);
|
2001-01-05 15:49:52 +03:00
|
|
|
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
|
2002-03-17 23:54:04 +03:00
|
|
|
i4b_l4_drvrdisc(int cdid)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
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;
|
2002-03-17 23:54:04 +03:00
|
|
|
md->header.cdid = cdid;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
i4bputqueue(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* send MSG_ACCT_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
2002-03-17 14:08:31 +03:00
|
|
|
i4b_l4_accounting(int cdid, int accttype, int ioutbytes,
|
2001-01-05 15:49:52 +03:00
|
|
|
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;
|
2002-03-17 14:08:31 +03:00
|
|
|
md->header.cdid = cdid;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
mp->controller = cd->isdnif;
|
2001-01-05 15:49:52 +03:00
|
|
|
mp->channel = cd->channelid;
|
|
|
|
mp->bprot = cd->bprot;
|
|
|
|
|
|
|
|
cd->dir = DIR_INCOMING;
|
|
|
|
|
|
|
|
if(strlen(cd->dst_telno) > 0)
|
2003-05-16 09:12:32 +04:00
|
|
|
strlcpy(mp->dst_telno, cd->dst_telno,
|
|
|
|
sizeof(mp->dst_telno));
|
2001-01-05 15:49:52 +03:00
|
|
|
else
|
2003-05-16 09:12:32 +04:00
|
|
|
strlcpy(mp->dst_telno, TELNO_EMPTY,
|
|
|
|
sizeof(mp->dst_telno));
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
if(strlen(cd->src_telno) > 0)
|
2003-05-16 09:12:32 +04:00
|
|
|
strlcpy(mp->src_telno, cd->src_telno,
|
|
|
|
sizeof(mp->src_telno));
|
2001-01-05 15:49:52 +03:00
|
|
|
else
|
2003-05-16 09:12:32 +04:00
|
|
|
strlcpy(mp->src_telno, TELNO_EMPTY,
|
|
|
|
sizeof(mp->src_telno));
|
2002-03-30 10:08:13 +03:00
|
|
|
mp->type_plan = cd->type_plan;
|
|
|
|
memcpy(mp->src_subaddr, cd->src_subaddr, sizeof(mp->src_subaddr));
|
|
|
|
memcpy(mp->dest_subaddr, cd->dest_subaddr, sizeof(mp->dest_subaddr));
|
2001-01-05 15:49:52 +03:00
|
|
|
|
2003-05-16 09:12:32 +04:00
|
|
|
strlcpy(mp->display, cd->display, sizeof(mp->src_telno));
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
mp->scr_ind = cd->scr_ind;
|
|
|
|
mp->prs_ind = cd->prs_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;
|
|
|
|
|
2001-01-19 15:44:44 +03:00
|
|
|
s = splnet();
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
cd->last_active_time = cd->connect_time = SECOND;
|
|
|
|
|
|
|
|
NDBGL4(L4_TIMO, "last_active/connect_time=%ld", (long)cd->connect_time);
|
|
|
|
|
|
|
|
i4b_link_bchandrvr(cd);
|
|
|
|
|
2002-05-02 22:56:55 +04:00
|
|
|
update_controller_leds(cd->l3drv);
|
|
|
|
|
2004-03-21 19:29:40 +03:00
|
|
|
if (cd->l4_driver != NULL && cd->l4_driver_softc != NULL)
|
|
|
|
(*cd->l4_driver->line_connected)(cd->l4_driver_softc, cd);
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
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;
|
2003-10-03 20:38:44 +04:00
|
|
|
mp->controller = cd->isdnif;
|
2001-01-05 15:49:52 +03:00
|
|
|
mp->channel = cd->channelid;
|
|
|
|
if(cd->datetime[0] != '\0')
|
2003-05-16 09:12:32 +04:00
|
|
|
strlcpy(mp->datetime, cd->datetime,
|
|
|
|
sizeof(mp->datetime));
|
2001-01-05 15:49:52 +03:00
|
|
|
else
|
|
|
|
mp->datetime[0] = '\0';
|
|
|
|
i4bputqueue(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* send MSG_DISCONNECT_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
i4b_l4_disconnect_ind(call_desc_t *cd)
|
|
|
|
{
|
2002-03-24 23:35:43 +03:00
|
|
|
struct isdn_l3_driver *d;
|
2001-01-05 15:49:52 +03:00
|
|
|
struct mbuf *m;
|
|
|
|
|
|
|
|
if(cd->timeout_active)
|
|
|
|
STOP_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd);
|
|
|
|
|
2002-03-17 23:54:04 +03:00
|
|
|
if (cd->l4_driver != NULL && cd->l4_driver_softc != NULL)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-17 23:54:04 +03:00
|
|
|
(*cd->l4_driver->line_disconnected)(cd->l4_driver_softc, (void *)cd);
|
2001-01-05 15:49:52 +03:00
|
|
|
i4b_unlink_bchandrvr(cd);
|
|
|
|
}
|
|
|
|
|
2002-03-30 14:43:33 +03:00
|
|
|
d = cd->l3drv;
|
2002-03-24 23:35:43 +03:00
|
|
|
|
2003-10-03 20:38:44 +04:00
|
|
|
if((cd->channelid >= 0) && (cd->channelid < d->nbch))
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-24 23:35:43 +03:00
|
|
|
d->bch_state[cd->channelid] = BCH_ST_FREE;
|
2003-09-25 18:17:57 +04:00
|
|
|
/*
|
|
|
|
* XXX: don't call l2 function for active cards.
|
|
|
|
*/
|
|
|
|
if (d->l3driver->N_DOWNLOAD == NULL)
|
|
|
|
i4b_l2_channel_set_state(d, cd->channelid, BCH_ST_FREE);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* no error, might be hunting call for callback */
|
2003-10-03 20:38:44 +04:00
|
|
|
NDBGL4(L4_MSG, "invalid channel %d for ISDN!", cd->channelid);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
2002-05-02 22:56:55 +04:00
|
|
|
update_controller_leds(d);
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
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;
|
2003-10-03 20:38:44 +04:00
|
|
|
mp->controller = cd->isdnif;
|
2001-01-05 15:49:52 +03:00
|
|
|
mp->channel = cd->channelid;
|
|
|
|
i4bputqueue(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* send MSG_PACKET_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
i4b_l4_packet_ind(int driver, int driver_unit, int dir, struct mbuf *pkt)
|
|
|
|
{
|
|
|
|
struct mbuf *m;
|
|
|
|
int len = pkt->m_pkthdr.len;
|
|
|
|
unsigned char *ip = pkt->m_data;
|
|
|
|
|
|
|
|
if((m = i4b_Dgetmbuf(sizeof(msg_packet_ind_t))) != NULL)
|
|
|
|
{
|
|
|
|
msg_packet_ind_t *mp = (msg_packet_ind_t *)m->m_data;
|
|
|
|
|
|
|
|
mp->header.type = MSG_PACKET_IND;
|
|
|
|
mp->header.cdid = -1;
|
|
|
|
mp->driver = driver;
|
|
|
|
mp->driver_unit = driver_unit;
|
|
|
|
mp->direction = dir;
|
|
|
|
memcpy(mp->pktdata, ip,
|
|
|
|
len <MAX_PACKET_LOG ? len : MAX_PACKET_LOG);
|
|
|
|
i4bputqueue(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-03-25 15:07:33 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* send MSG_CONTR_EV_IND message to userland
|
|
|
|
*---------------------------------------------------------------------------*/
|
2002-03-29 23:29:53 +03:00
|
|
|
void
|
2002-03-25 15:07:33 +03:00
|
|
|
i4b_l4_contr_ev_ind(int controller, int attach)
|
|
|
|
{
|
|
|
|
struct mbuf *m;
|
|
|
|
|
|
|
|
if((m = i4b_Dgetmbuf(sizeof(msg_ctrl_ev_ind_t))) != NULL)
|
|
|
|
{
|
|
|
|
msg_ctrl_ev_ind_t *ev = (msg_ctrl_ev_ind_t *)m->m_data;
|
|
|
|
|
|
|
|
ev->header.type = MSG_CONTR_EV_IND;
|
|
|
|
ev->header.cdid = -1;
|
|
|
|
ev->controller = controller;
|
|
|
|
ev->event = attach;
|
|
|
|
i4bputqueue(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-01-05 15:49:52 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* link a driver(unit) to a B-channel(controller,unit,channel)
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static int
|
|
|
|
i4b_link_bchandrvr(call_desc_t *cd)
|
|
|
|
{
|
2002-03-30 14:43:33 +03:00
|
|
|
struct isdn_l3_driver *d = cd->l3drv;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
if (d == NULL || d->l3driver == NULL || d->l3driver->get_linktab == NULL)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
cd->ilt = NULL;
|
2002-03-24 23:35:43 +03:00
|
|
|
return 1;
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
cd->ilt = d->l3driver->get_linktab(d->l1_token,
|
|
|
|
cd->channelid);
|
|
|
|
|
2002-03-17 23:54:04 +03:00
|
|
|
cd->l4_driver = isdn_l4_get_driver(cd->bchan_driver_index, cd->bchan_driver_unit);
|
|
|
|
if (cd->l4_driver != NULL)
|
|
|
|
cd->l4_driver_softc = cd->l4_driver->get_softc(cd->bchan_driver_unit);
|
|
|
|
else
|
|
|
|
cd->l4_driver_softc = NULL;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
2002-03-17 23:54:04 +03:00
|
|
|
if(cd->l4_driver == NULL || cd->l4_driver_softc == NULL || cd->ilt == NULL)
|
2001-01-05 15:49:52 +03:00
|
|
|
return(-1);
|
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
if (d->l3driver->set_l4_driver != NULL)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-24 23:35:43 +03:00
|
|
|
d->l3driver->set_l4_driver(d->l1_token,
|
|
|
|
cd->channelid, cd->l4_driver, cd->l4_driver_softc);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
2002-03-17 23:54:04 +03:00
|
|
|
cd->l4_driver->set_linktab(cd->l4_driver_softc, cd->ilt);
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* activate B channel */
|
|
|
|
|
2002-03-17 12:45:58 +03:00
|
|
|
(*cd->ilt->bchannel_driver->bch_config)(cd->ilt->l1token, cd->ilt->channel, cd->bprot, 1);
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* unlink a driver(unit) from a B-channel(controller,unit,channel)
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
i4b_unlink_bchandrvr(call_desc_t *cd)
|
|
|
|
{
|
2002-03-30 14:43:33 +03:00
|
|
|
struct isdn_l3_driver *d = cd->l3drv;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
/*
|
|
|
|
* XXX - what's this *cd manipulation for? Shouldn't we
|
|
|
|
* close the bchannel driver first and then just set ilt to NULL
|
|
|
|
* in *cd?
|
|
|
|
*/
|
|
|
|
if (d == NULL || d->l3driver == NULL || d->l3driver->get_linktab == NULL)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
cd->ilt = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2002-03-24 23:35:43 +03:00
|
|
|
cd->ilt = d->l3driver->get_linktab(
|
|
|
|
d->l1_token, cd->channelid);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* deactivate B channel */
|
|
|
|
|
2002-03-17 12:45:58 +03:00
|
|
|
(*cd->ilt->bchannel_driver->bch_config)(cd->ilt->l1token, cd->ilt->channel, cd->bprot, 0);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
---------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
static time_t
|
|
|
|
i4b_get_idletime(call_desc_t *cd)
|
|
|
|
{
|
2004-03-21 19:29:40 +03:00
|
|
|
if (cd->l4_driver != NULL && cd->l4_driver_softc != NULL
|
|
|
|
&& cd->l4_driver->get_idletime)
|
2002-03-17 23:54:04 +03:00
|
|
|
return cd->l4_driver->get_idletime(cd->l4_driver_softc);
|
|
|
|
return cd->last_active_time;
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
2002-03-17 23:54:04 +03:00
|
|
|
|
2001-01-05 15:49:52 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* B channel idle check timeout setup
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
i4b_l4_setup_timeout(call_desc_t *cd)
|
|
|
|
{
|
|
|
|
NDBGL4(L4_TIMO, "%ld: direction %d, shorthold algorithm %d",
|
|
|
|
(long)SECOND, cd->dir, cd->shorthold_data.shorthold_algorithm);
|
|
|
|
|
|
|
|
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 */
|
|
|
|
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz/2);
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
NDBGL4(L4_TIMO, "%ld: incoming-call, setup max_idle_time to %ld", (long)SECOND, (long)cd->max_idle_time);
|
|
|
|
}
|
|
|
|
else if((cd->dir == DIR_OUTGOING) && (cd->shorthold_data.idle_time > 0))
|
|
|
|
{
|
|
|
|
switch( cd->shorthold_data.shorthold_algorithm )
|
|
|
|
{
|
|
|
|
default: /* fall into the old fix algorithm */
|
|
|
|
case SHA_FIXU:
|
|
|
|
i4b_l4_setup_timeout_fix_unit( cd );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SHA_VARU:
|
|
|
|
i4b_l4_setup_timeout_var_unit( cd );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NDBGL4(L4_TIMO, "no idle_timeout configured");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* fixed unit algorithm B channel idle check timeout setup
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
i4b_l4_setup_timeout_fix_unit(call_desc_t *cd)
|
|
|
|
{
|
|
|
|
/* outgoing call */
|
|
|
|
|
|
|
|
if((cd->shorthold_data.idle_time > 0) && (cd->shorthold_data.unitlen_time == 0))
|
|
|
|
{
|
|
|
|
/* outgoing call: simple max idletime check */
|
|
|
|
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz/2);
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, setup idle_time to %ld",
|
|
|
|
(long)SECOND, (long)cd->shorthold_data.idle_time);
|
|
|
|
}
|
|
|
|
else if((cd->shorthold_data.unitlen_time > 0) && (cd->shorthold_data.unitlen_time > (cd->shorthold_data.idle_time + cd->shorthold_data.earlyhup_time)))
|
|
|
|
{
|
|
|
|
/* outgoing call: full shorthold mode check */
|
|
|
|
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz*(cd->shorthold_data.unitlen_time - (cd->shorthold_data.idle_time + cd->shorthold_data.earlyhup_time)));
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
cd->idletime_state = IST_NONCHK;
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, start %ld sec nocheck window",
|
|
|
|
(long)SECOND, (long)(cd->shorthold_data.unitlen_time - (cd->shorthold_data.idle_time + cd->shorthold_data.earlyhup_time)));
|
|
|
|
|
|
|
|
if(cd->aocd_flag == 0)
|
|
|
|
{
|
|
|
|
cd->units_type = CHARGE_CALC;
|
|
|
|
cd->cunits++;
|
|
|
|
i4b_l4_charging_ind(cd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* parms somehow got wrong .. */
|
|
|
|
|
|
|
|
NDBGL4(L4_ERR, "%ld: ERROR: idletime[%ld]+earlyhup[%ld] > unitlength[%ld]!",
|
|
|
|
(long)SECOND, (long)cd->shorthold_data.idle_time, (long)cd->shorthold_data.earlyhup_time, (long)cd->shorthold_data.unitlen_time);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* variable unit algorithm B channel idle check timeout setup
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
i4b_l4_setup_timeout_var_unit(call_desc_t *cd)
|
|
|
|
{
|
|
|
|
/* outgoing call: variable unit idletime check */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* start checking for an idle connect one second before the end of the unit.
|
|
|
|
* The one second takes into account of rounding due to the driver only
|
|
|
|
* using the seconds and not the uSeconds of the current time
|
|
|
|
*/
|
|
|
|
cd->idletime_state = IST_CHECK; /* move directly to the checking state */
|
|
|
|
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz * (cd->shorthold_data.unitlen_time - 1) );
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, var idle time - setup to %ld",
|
|
|
|
(long)SECOND, (long)cd->shorthold_data.unitlen_time);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* B channel idle check timeout function
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
i4b_idle_check(call_desc_t *cd)
|
|
|
|
{
|
|
|
|
int s;
|
|
|
|
|
|
|
|
if(cd->cdid == CDID_UNUSED)
|
|
|
|
return;
|
|
|
|
|
2001-01-19 15:44:44 +03:00
|
|
|
s = splnet();
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* failsafe */
|
|
|
|
|
|
|
|
if(cd->timeout_active == 0)
|
|
|
|
{
|
|
|
|
NDBGL4(L4_ERR, "ERROR: timeout_active == 0 !!!");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cd->timeout_active = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* incoming connections, simple idletime check */
|
|
|
|
|
|
|
|
if(cd->dir == DIR_INCOMING)
|
|
|
|
{
|
|
|
|
if((i4b_get_idletime(cd) + cd->max_idle_time) <= SECOND)
|
|
|
|
{
|
2002-03-30 14:43:33 +03:00
|
|
|
struct isdn_l3_driver *d = cd->l3drv;
|
2001-01-05 15:49:52 +03:00
|
|
|
NDBGL4(L4_TIMO, "%ld: incoming-call, line idle timeout, disconnecting!", (long)SECOND);
|
2002-03-30 14:15:41 +03:00
|
|
|
d->l3driver->N_DISCONNECT_REQUEST(cd,
|
2001-01-05 15:49:52 +03:00
|
|
|
(CAUSET_I4B << 8) | CAUSE_I4B_NORMAL);
|
|
|
|
i4b_l4_idle_timeout_ind(cd);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NDBGL4(L4_TIMO, "%ld: incoming-call, activity, last_active=%ld, max_idle=%ld", (long)SECOND, (long)i4b_get_idletime(cd), (long)cd->max_idle_time);
|
|
|
|
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz/2);
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* outgoing connections */
|
|
|
|
|
|
|
|
else if(cd->dir == DIR_OUTGOING)
|
|
|
|
{
|
|
|
|
switch( cd->shorthold_data.shorthold_algorithm )
|
|
|
|
{
|
|
|
|
case SHA_FIXU:
|
|
|
|
i4b_idle_check_fix_unit( cd );
|
|
|
|
break;
|
|
|
|
case SHA_VARU:
|
|
|
|
i4b_idle_check_var_unit( cd );
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
NDBGL4(L4_TIMO, "%ld: bad value for shorthold_algorithm of %d",
|
|
|
|
(long)SECOND, cd->shorthold_data.shorthold_algorithm);
|
|
|
|
i4b_idle_check_fix_unit( cd );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
splx(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* fixed unit algorithm B channel idle check timeout function
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
i4b_idle_check_fix_unit(call_desc_t *cd)
|
|
|
|
{
|
2002-03-30 14:43:33 +03:00
|
|
|
struct isdn_l3_driver *d = cd->l3drv;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* simple idletime calculation */
|
|
|
|
|
|
|
|
if((cd->shorthold_data.idle_time > 0) && (cd->shorthold_data.unitlen_time == 0))
|
|
|
|
{
|
|
|
|
if((i4b_get_idletime(cd) + cd->shorthold_data.idle_time) <= SECOND)
|
|
|
|
{
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call-st, idle timeout, disconnecting!", (long)SECOND);
|
2002-03-30 14:15:41 +03:00
|
|
|
d->l3driver->N_DISCONNECT_REQUEST(cd, (CAUSET_I4B << 8) | CAUSE_I4B_NORMAL);
|
2001-01-05 15:49:52 +03:00
|
|
|
i4b_l4_idle_timeout_ind(cd);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call-st, activity, last_active=%ld, max_idle=%ld",
|
|
|
|
(long)SECOND, (long)i4b_get_idletime(cd), (long)cd->shorthold_data.idle_time);
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz/2);
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* full shorthold mode calculation */
|
|
|
|
|
|
|
|
else if((cd->shorthold_data.unitlen_time > 0)
|
|
|
|
&& (cd->shorthold_data.unitlen_time > (cd->shorthold_data.idle_time + cd->shorthold_data.earlyhup_time)))
|
|
|
|
{
|
|
|
|
switch(cd->idletime_state)
|
|
|
|
{
|
|
|
|
|
|
|
|
case IST_NONCHK: /* end of non-check time */
|
|
|
|
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz*(cd->shorthold_data.idle_time));
|
|
|
|
cd->idletimechk_start = SECOND;
|
|
|
|
cd->idletime_state = IST_CHECK;
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, idletime check window reached!", (long)SECOND);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IST_CHECK: /* end of idletime chk */
|
|
|
|
if((i4b_get_idletime(cd) > cd->idletimechk_start) &&
|
|
|
|
(i4b_get_idletime(cd) <= SECOND))
|
|
|
|
{ /* activity detected */
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz*(cd->shorthold_data.earlyhup_time));
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
cd->idletime_state = IST_SAFE;
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, activity at %ld, wait earlyhup-end", (long)SECOND, (long)i4b_get_idletime(cd));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ /* no activity, hangup */
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, idle timeout, last activity at %ld", (long)SECOND, (long)i4b_get_idletime(cd));
|
2002-03-30 14:15:41 +03:00
|
|
|
d->l3driver->N_DISCONNECT_REQUEST(cd, (CAUSET_I4B << 8) | CAUSE_I4B_NORMAL);
|
2001-01-05 15:49:52 +03:00
|
|
|
i4b_l4_idle_timeout_ind(cd);
|
|
|
|
cd->idletime_state = IST_IDLE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IST_SAFE: /* end of earlyhup time */
|
|
|
|
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz*(cd->shorthold_data.unitlen_time - (cd->shorthold_data.idle_time+cd->shorthold_data.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);
|
|
|
|
}
|
|
|
|
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, earlyhup end, wait for idletime start", (long)SECOND);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
NDBGL4(L4_ERR, "outgoing-call: invalid idletime_state value!");
|
|
|
|
cd->idletime_state = IST_IDLE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* variable unit algorithm B channel idle check timeout function
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
i4b_idle_check_var_unit(call_desc_t *cd)
|
|
|
|
{
|
|
|
|
switch(cd->idletime_state)
|
|
|
|
{
|
|
|
|
|
|
|
|
/* see if there has been any activity within the last idle_time seconds */
|
|
|
|
case IST_CHECK:
|
|
|
|
if( i4b_get_idletime(cd) > (SECOND - cd->shorthold_data.idle_time))
|
|
|
|
{ /* activity detected */
|
|
|
|
#if defined(__FreeBSD_version) && __FreeBSD_version >= 300001
|
|
|
|
cd->idle_timeout_handle =
|
|
|
|
#endif
|
|
|
|
/* check again in one second */
|
|
|
|
START_TIMER(cd->idle_timeout_handle, i4b_idle_check, cd, hz);
|
|
|
|
cd->timeout_active = 1;
|
|
|
|
cd->idletime_state = IST_CHECK;
|
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, var idle timeout - activity at %ld, continuing", (long)SECOND, (long)i4b_get_idletime(cd));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ /* no activity, hangup */
|
2002-03-30 14:43:33 +03:00
|
|
|
struct isdn_l3_driver *d = cd->l3drv;
|
2001-01-05 15:49:52 +03:00
|
|
|
NDBGL4(L4_TIMO, "%ld: outgoing-call, var idle timeout - last activity at %ld", (long)SECOND, (long)i4b_get_idletime(cd));
|
2002-03-30 14:15:41 +03:00
|
|
|
d->l3driver->N_DISCONNECT_REQUEST(cd, (CAUSET_I4B << 8) | CAUSE_I4B_NORMAL);
|
2001-01-05 15:49:52 +03:00
|
|
|
i4b_l4_idle_timeout_ind(cd);
|
|
|
|
cd->idletime_state = IST_IDLE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
NDBGL4(L4_ERR, "outgoing-call: var idle timeout invalid idletime_state value!");
|
|
|
|
cd->idletime_state = IST_IDLE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-03-16 19:55:51 +03:00
|
|
|
#endif /* NISDN > 0 */
|