2002-04-14 16:24:26 +04:00
|
|
|
/*-
|
|
|
|
* Copyright (c) 2002 The NetBSD Foundation, Inc.
|
2001-01-05 15:49:52 +03:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
2002-04-14 16:24:26 +04:00
|
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
2003-12-04 16:57:30 +03:00
|
|
|
* by Martin Husemann <martin@NetBSD.org>.
|
2002-04-14 16:24:26 +04:00
|
|
|
*
|
2001-01-05 15:49:52 +03:00
|
|
|
* 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.
|
2002-04-14 16:24:26 +04:00
|
|
|
* 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.
|
2001-01-05 15:49:52 +03:00
|
|
|
*
|
2002-04-14 16:24:26 +04:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. 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 FOUNDATION 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.
|
2001-01-05 15:49:52 +03:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2002-04-14 16:24:26 +04:00
|
|
|
#include <sys/cdefs.h>
|
2015-08-30 12:46:57 +03:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: daic.c,v 1.32 2015/08/30 09:46:57 martin Exp $");
|
2002-04-14 16:24:26 +04:00
|
|
|
|
2001-01-05 15:49:52 +03:00
|
|
|
/*
|
|
|
|
* daic.c: MI driver for Diehl active ISDN cards (S, SX, SXn, SCOM, QUADRO)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/errno.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/device.h>
|
|
|
|
#include <sys/malloc.h>
|
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <net/if.h>
|
|
|
|
|
|
|
|
#include <netisdn/i4b_ioctl.h>
|
|
|
|
#include <netisdn/i4b_l3l4.h>
|
|
|
|
#include <netisdn/i4b_isdnq931.h>
|
|
|
|
#include <netisdn/i4b_q931.h>
|
|
|
|
#include <netisdn/i4b_l3fsm.h>
|
|
|
|
#include <netisdn/i4b_l4.h>
|
|
|
|
|
2007-10-19 15:59:34 +04:00
|
|
|
#include <sys/bus.h>
|
2001-01-05 15:49:52 +03:00
|
|
|
#include <dev/ic/daicvar.h>
|
|
|
|
#include <dev/ic/daicreg.h>
|
|
|
|
#include <dev/microcode/daic/dnload.h>
|
|
|
|
|
|
|
|
#ifdef NetBSD1_3
|
|
|
|
#if NetBSD1_3 < 2
|
|
|
|
/* the device is MI, only the attach struct is in the bus
|
2003-11-02 14:07:44 +03:00
|
|
|
dependent frontend. And only on old versions... */
|
2001-01-05 15:49:52 +03:00
|
|
|
struct cfdriver daic_cd = {
|
|
|
|
NULL, "daic", DV_DULL
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* local function prototypes */
|
2007-02-26 00:06:57 +03:00
|
|
|
static const char * cardtypename(int cardtype);
|
2005-02-04 05:10:35 +03:00
|
|
|
static int daic_download(void *, int portcount, struct isdn_dr_prot *data);
|
|
|
|
static int daic_diagnostic(void *, struct isdn_diagnostic_request *req);
|
2002-03-30 14:15:41 +03:00
|
|
|
static void daic_connect_request(struct call_desc *cd);
|
|
|
|
static void daic_connect_response(struct call_desc *cd, int, int);
|
|
|
|
static void daic_disconnect_request(struct call_desc *cd, int);
|
2005-02-04 05:10:35 +03:00
|
|
|
static int daic_reset(bus_space_tag_t bus, bus_space_handle_t io, int port, int *memsize);
|
|
|
|
static int daic_handle_intr(struct daic_softc *sc, int port);
|
2002-03-24 23:35:43 +03:00
|
|
|
static void daic_register_port(struct daic_softc *sc, int port);
|
2002-03-22 12:54:15 +03:00
|
|
|
static void daic_request(struct daic_softc *sc, int port, u_int req, u_int id, bus_size_t parmsize, const u_int8_t *parms);
|
|
|
|
static u_int daic_assign(struct daic_softc *sc, int port, u_int instance, bus_size_t parmsize, const u_int8_t *parms);
|
|
|
|
static void daic_indicate_ind(struct daic_softc *sc, int port);
|
|
|
|
static void daic_bch_config(void *, int channel, int bprot, int updown);
|
|
|
|
static void daic_bch_tx_start(void *, int channel);
|
|
|
|
static void daic_set_link(void *softc, int channel,
|
|
|
|
const struct isdn_l4_driver_functions *l4_driver, void *l4_inst );
|
2002-03-30 14:15:41 +03:00
|
|
|
static void daic_mgmt_command(struct isdn_l3_driver *drv, int cmd, void *parm);
|
|
|
|
static void daic_alert_request(struct call_desc *cd);
|
2002-03-22 12:54:15 +03:00
|
|
|
|
|
|
|
static isdn_link_t *daic_ret_linktab(void *softc, int channel);
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
#ifdef DAIC_DEBUG
|
2002-03-22 12:54:15 +03:00
|
|
|
static void daic_dump_request(struct daic_softc *sc, int port, u_int req, u_int id, bus_size_t parmsize, u_int8_t *parms);
|
2001-01-05 15:49:52 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* static data */
|
2007-02-26 00:06:57 +03:00
|
|
|
static const char * const cardnames[] = {
|
2001-01-05 15:49:52 +03:00
|
|
|
"S", "SX", "SCOM", "QUADRO"
|
|
|
|
};
|
|
|
|
|
2007-02-26 00:06:57 +03:00
|
|
|
static const char * const err_codes[DAIC_RC_ERRMASK+1] = {
|
2001-01-05 15:49:52 +03:00
|
|
|
"NO ERROR",
|
|
|
|
"UNKNOWN COMMAND",
|
|
|
|
"WRONG COMMAND",
|
|
|
|
"WRONG ID",
|
|
|
|
"WRONG CH",
|
|
|
|
"UNKNOWN IE",
|
|
|
|
"WRONG IE",
|
|
|
|
"OUT OF RESOURCES"
|
|
|
|
};
|
|
|
|
|
|
|
|
/* fixed parameters */
|
|
|
|
|
|
|
|
/* no parameters */
|
|
|
|
static u_int8_t parm_none[] = { 0 };
|
|
|
|
#define VOIDREQ(sc,port,req,id) daic_request(sc, port, req, id, sizeof parm_none, parm_none)
|
|
|
|
|
|
|
|
/* assign request for the global d-channel instance */
|
|
|
|
static u_int8_t parm_global_assign[] = {
|
2003-04-06 22:20:07 +04:00
|
|
|
/* BC len cap rate A-law */
|
2001-01-05 15:49:52 +03:00
|
|
|
0x04, 0x03, 0x80, 0x90, 0xa3, /* 64k speech */
|
|
|
|
0x04, 0x02, 0x88, 0x90, /* 64k data */
|
|
|
|
0x04, 0x03, 0x89, 0x90, 0xa3, /* restricted digital info */
|
|
|
|
0x04, 0x03, 0x90, 0x90, 0xa3, /* 3.1k speech */
|
|
|
|
/* shift6 SIN len service */
|
|
|
|
0x96, 0x01, 0x02, 0x00, 0x00, /* any service */
|
|
|
|
/* end of parms */
|
|
|
|
0x00
|
|
|
|
};
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Return the name of a card with given cardtype
|
|
|
|
*---------------------------------------------------------------------------*/
|
2007-02-26 00:06:57 +03:00
|
|
|
static const char *
|
2009-03-14 18:35:58 +03:00
|
|
|
cardtypename(int cardtype)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
if (cardtype >= 0 && cardtype < (sizeof(cardnames) / sizeof(cardnames[0])))
|
|
|
|
return cardnames[cardtype];
|
|
|
|
else
|
|
|
|
return "unknown type";
|
|
|
|
}
|
2005-02-27 03:26:58 +03:00
|
|
|
|
2001-01-05 15:49:52 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
2005-02-27 03:26:58 +03:00
|
|
|
* Probe for presence of device at given io space.
|
|
|
|
* Return the card type (stupid ISA needs to know this in advance, to
|
2002-03-25 12:08:09 +03:00
|
|
|
* calculate the share memory size).
|
2001-01-05 15:49:52 +03:00
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
int
|
2009-03-14 18:35:58 +03:00
|
|
|
daic_probe(bus_space_tag_t bus, bus_space_handle_t io)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-25 12:08:09 +03:00
|
|
|
return (daic_reset(bus, io, 0, NULL));
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Attach and initialize the card at given io space.
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
2009-05-12 18:16:35 +04:00
|
|
|
daic_attach(device_t self, struct daic_softc *sc)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-22 12:54:15 +03:00
|
|
|
int i, num_ports, memsize = 0;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* init sc */
|
2002-03-22 12:54:15 +03:00
|
|
|
memset(sc->sc_port, 0, sizeof sc->sc_port);
|
|
|
|
memset(sc->sc_con, 0, sizeof sc->sc_con);
|
2001-01-05 15:49:52 +03:00
|
|
|
sc->sc_cardtype = -1;
|
|
|
|
|
|
|
|
/* init card */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_cardtype = daic_reset(sc->sc_iot, sc->sc_ioh, 0, &memsize);
|
|
|
|
if (sc->sc_cardtype == 0) {
|
|
|
|
printf(": unknown card, can not attach.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
printf("\n");
|
2012-10-27 21:17:22 +04:00
|
|
|
printf("%s: EICON.Diehl %s\n", device_xname(sc->sc_dev),
|
2002-03-24 23:35:43 +03:00
|
|
|
cardtypename(sc->sc_cardtype));
|
2012-10-27 21:17:22 +04:00
|
|
|
printf("%s: %d kByte on board RAM\n", device_xname(sc->sc_dev), memsize);
|
2002-03-22 12:54:15 +03:00
|
|
|
num_ports = sc->sc_cardtype == DAIC_TYPE_QUAD ? 4 : 1;
|
|
|
|
for (i = 0; i < num_ports; i++)
|
|
|
|
sc->sc_port[i].du_state = DAIC_STATE_DOWNLOAD;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* register all ports this card has */
|
|
|
|
for (i = 0; i < num_ports; i++)
|
|
|
|
daic_register_port(sc, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* handle interrupts for one port of the card
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static int
|
2009-03-14 18:35:58 +03:00
|
|
|
daic_handle_intr(struct daic_softc *sc, int port)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
struct outcallentry *assoc;
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_unit * du = &sc->sc_port[port];
|
2001-01-05 15:49:52 +03:00
|
|
|
int off = port * DAIC_ISA_MEMSIZE;
|
|
|
|
u_int8_t rc, rcid;
|
|
|
|
u_int8_t ind, indid;
|
|
|
|
int chan;
|
|
|
|
|
|
|
|
/* check if we caused the interrupt */
|
|
|
|
if (!bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_IRQ+off))
|
|
|
|
return 0; /* nope, exit */
|
|
|
|
|
|
|
|
/* is the card in running state yet? */
|
2002-03-22 12:54:15 +03:00
|
|
|
if (du->du_state == DAIC_STATE_TESTING) {
|
2001-01-05 15:49:52 +03:00
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RC+off, 0);
|
2002-03-22 12:54:15 +03:00
|
|
|
du->du_state = DAIC_STATE_RUNNING;
|
|
|
|
wakeup(du);
|
2001-01-05 15:49:52 +03:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* what caused the interrupt? */
|
|
|
|
/* (1) Check for a return code */
|
|
|
|
rc = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RC+off);
|
|
|
|
rcid = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RCID+off);
|
|
|
|
if (!rc) goto check_ind;
|
|
|
|
|
|
|
|
/* maybe an assign answer (positive or negative) */
|
|
|
|
if (rc == DAIC_RC_ASSIGN_OK) {
|
2005-02-27 03:26:58 +03:00
|
|
|
du->du_assign_res = rcid;
|
2001-01-05 15:49:52 +03:00
|
|
|
/* assing rc is special, we tell the card it's done */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_REQ+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RC+off, 0);
|
|
|
|
/* we handle some types of assigns to global dchannel id's automaticaly */
|
2002-03-22 12:54:15 +03:00
|
|
|
if (du->du_assign & DAIC_ASSIGN_GLOBAL) {
|
|
|
|
du->du_global_dchan = rcid;
|
|
|
|
du->du_assign &= ~(DAIC_ASSIGN_GLOBAL|DAIC_ASSIGN_PENDING);
|
|
|
|
if (du->du_assign & DAIC_ASSIGN_SLEEPING) {
|
|
|
|
du->du_assign = 0;
|
|
|
|
wakeup(&du->du_assign_res);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
} else {
|
2002-03-22 12:54:15 +03:00
|
|
|
wakeup(&du->du_assign);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
goto check_ind;
|
|
|
|
} else if ((rc & DAIC_RC_ASSIGN_MASK) == DAIC_RC_ASSIGN_RC) {
|
2012-10-27 21:17:22 +04:00
|
|
|
aprint_error_dev(sc->sc_dev, "assign request failed, error 0x%02x: %s\n",
|
2008-04-08 16:07:25 +04:00
|
|
|
rc & DAIC_RC_ERRMASK,
|
2001-01-05 15:49:52 +03:00
|
|
|
err_codes[rc & DAIC_RC_ERRMASK]);
|
2002-03-22 12:54:15 +03:00
|
|
|
du->du_assign_res = 0;
|
2001-01-05 15:49:52 +03:00
|
|
|
/* assing rc is special, we tell the card it's done */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_REQ+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RC+off, 0);
|
|
|
|
/* that's it */
|
2002-03-22 12:54:15 +03:00
|
|
|
wakeup(&du->du_assign);
|
2001-01-05 15:49:52 +03:00
|
|
|
goto check_ind;
|
|
|
|
}
|
2002-03-22 12:54:15 +03:00
|
|
|
if (rcid == du->du_global_dchan) {
|
|
|
|
du->du_request_res = rc;
|
|
|
|
wakeup(&du->du_request_res);
|
2001-01-05 15:49:52 +03:00
|
|
|
goto req_done;
|
|
|
|
}
|
|
|
|
for (chan = 0; chan < 2; chan++) {
|
|
|
|
if (rcid == sc->sc_con[port*2+chan].dchan_inst) {
|
|
|
|
sc->sc_con[port*2+chan].dchan_rc = rc;
|
|
|
|
wakeup(&sc->sc_con[port*2+chan].dchan_rc);
|
|
|
|
goto req_done;
|
|
|
|
} else if (rcid == sc->sc_con[port*2+chan].bchan_inst) {
|
|
|
|
sc->sc_con[port*2+chan].bchan_rc = rc;
|
|
|
|
wakeup(&sc->sc_con[port*2+chan].bchan_rc);
|
|
|
|
goto req_done;
|
|
|
|
}
|
|
|
|
}
|
2002-03-22 12:54:15 +03:00
|
|
|
TAILQ_FOREACH(assoc, &sc->sc_outcalls[port], queue) {
|
2001-01-05 15:49:52 +03:00
|
|
|
if (rcid == assoc->dchan_id) {
|
|
|
|
assoc->rc = rc;
|
|
|
|
wakeup(assoc);
|
|
|
|
goto req_done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* not found? */
|
|
|
|
printf("%s: unknown id 0x%02x got rc 0x%02x: %s\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), rcid, rc,
|
2001-01-05 15:49:52 +03:00
|
|
|
err_codes[rc & DAIC_RC_ERRMASK]);
|
|
|
|
|
|
|
|
req_done:
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RC+off, 0);
|
|
|
|
|
|
|
|
check_ind:
|
|
|
|
/* (2) Check for an indication */
|
|
|
|
ind = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_IND+off);
|
|
|
|
indid = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_INDID+off);
|
|
|
|
if (!ind) goto done;
|
|
|
|
|
|
|
|
/* incoming call routed to global dchannel task? */
|
2002-03-22 12:54:15 +03:00
|
|
|
if (indid == du->du_global_dchan) {
|
2001-01-05 15:49:52 +03:00
|
|
|
if (ind == DAIC_IND_INDICATE) {
|
|
|
|
daic_indicate_ind(sc, port);
|
|
|
|
} else if (ind == DAIC_IND_INFO) {
|
|
|
|
int i;
|
|
|
|
|
2005-02-27 03:26:58 +03:00
|
|
|
printf("%s: got info indication\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev));
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
for (i = 0; i < 48; i++) {
|
|
|
|
if (!(i % 16))
|
|
|
|
printf("\n%02x:", i);
|
|
|
|
printf(" %02x", bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RBUFFER+off+i));
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
} else if (ind == DAIC_IND_HANGUP) {
|
2005-02-27 03:26:58 +03:00
|
|
|
printf("%s: got global HANGUP indication\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev));
|
2001-01-05 15:49:52 +03:00
|
|
|
} else {
|
2005-02-27 03:26:58 +03:00
|
|
|
printf("%s: unknown global indication: 0x%02x\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), ind);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
goto ind_done;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (chan = 0; chan < 2; chan++) {
|
|
|
|
if (indid == sc->sc_con[port*2+chan].dchan_inst) {
|
|
|
|
printf("%s: D-Channel indication 0x%02x for channel %d\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), ind, chan);
|
2001-01-05 15:49:52 +03:00
|
|
|
goto ind_done;
|
|
|
|
} else if (indid == sc->sc_con[port*2+chan].bchan_inst) {
|
|
|
|
printf("%s: B-Channel indication 0x%02x for channel %d\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), ind, chan);
|
2001-01-05 15:49:52 +03:00
|
|
|
goto ind_done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-03-22 12:54:15 +03:00
|
|
|
TAILQ_FOREACH(assoc, &sc->sc_outcalls[port], queue) {
|
2001-01-05 15:49:52 +03:00
|
|
|
if (indid == assoc->dchan_id) {
|
|
|
|
printf("%s: D-Channel indication 0x%02x for outgoing call with cdid %d\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), ind, assoc->cdid);
|
2001-01-05 15:49:52 +03:00
|
|
|
goto ind_done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* not found - something's wrong! */
|
2012-10-27 21:17:22 +04:00
|
|
|
printf("%s: got ind 0x%02x for id 0x%02x\n", device_xname(sc->sc_dev), ind, indid);
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
ind_done:
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_IND+off, 0);
|
|
|
|
|
|
|
|
done:
|
|
|
|
/* tell card we're ready for more... */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_IRQ+off, 0);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Handle interrupts
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
int
|
2009-03-14 18:35:58 +03:00
|
|
|
daic_intr(struct daic_softc *sc)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
int handeld = 0;
|
|
|
|
if (sc->sc_cardtype == DAIC_TYPE_QUAD) {
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
handeld |= daic_handle_intr(sc, i);
|
|
|
|
} else
|
|
|
|
handeld = daic_handle_intr(sc, 0);
|
|
|
|
return handeld;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Download primary protocol microcode to on-board processor
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static int
|
2009-03-14 18:35:58 +03:00
|
|
|
daic_download(void *token, int count, struct isdn_dr_prot *data)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_unit *du = token;
|
|
|
|
struct daic_softc *sc = du->du_sc;
|
2001-01-05 15:49:52 +03:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if (sc->sc_cardtype != DAIC_TYPE_QUAD)
|
|
|
|
count = 1; /* XXX - or signal error ? */
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
int off = DAIC_ISA_MEMSIZE * i;
|
|
|
|
u_int8_t *p = data[i].microcode;
|
|
|
|
size_t s = data[i].bytecount;
|
|
|
|
u_int32_t sw_id;
|
|
|
|
int cnt, x;
|
|
|
|
for (p = data[i].microcode+4, cnt = 0; *p && cnt < 70; p++, cnt++)
|
|
|
|
;
|
|
|
|
sw_id = p[1] | (p[2] << 8) | (p[3] << 16) | (p[4] << 24);
|
|
|
|
if (sc->sc_cardtype == DAIC_TYPE_QUAD)
|
|
|
|
printf("%s port %d: downloading %s\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), i, data[i].microcode+4);
|
2001-01-05 15:49:52 +03:00
|
|
|
else
|
|
|
|
printf("%s: downloading %s\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), data[i].microcode+4);
|
2001-01-05 15:49:52 +03:00
|
|
|
x = splnet();
|
|
|
|
p = data[i].microcode;
|
|
|
|
while (s > 0) {
|
|
|
|
size_t size = (s > 256) ? 256 : s;
|
|
|
|
bus_space_write_region_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_BUF+off, p, size);
|
|
|
|
p += size;
|
|
|
|
s -= size;
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_CTRL+off, 1);
|
|
|
|
splx(x);
|
|
|
|
for (cnt = 0; cnt < 2*hz; cnt++) {
|
|
|
|
x = splnet();
|
|
|
|
if (bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_CTRL+off) == 0)
|
|
|
|
break;
|
|
|
|
splx(x);
|
|
|
|
tsleep(sc, 0, "daic download", 1);
|
|
|
|
}
|
|
|
|
if (bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_CTRL+off) != 0) {
|
2012-10-27 21:17:22 +04:00
|
|
|
aprint_error_dev(sc->sc_dev, "download of microcode failed\n");
|
2001-01-05 15:49:52 +03:00
|
|
|
return EIO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* configure microcode - no parameters yet - XXX */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_TEI+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_NT2+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_ZERO+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_WATCHDOG+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_PERMANENT+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_XINTERFACE+off, 0);
|
|
|
|
|
|
|
|
/* start protocol */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_CTRL+off, 2);
|
|
|
|
|
|
|
|
/* wait for signature */
|
|
|
|
for (cnt = 0; cnt < 2*hz; cnt++) {
|
|
|
|
u_int16_t signature;
|
|
|
|
signature = bus_space_read_2(sc->sc_iot, sc->sc_ioh, DAIC_BOOT_SIGNATURE+off);
|
|
|
|
if (signature == DAIC_SIGNATURE_VALUE)
|
|
|
|
break;
|
|
|
|
if (signature) {
|
|
|
|
if (signature != DAIC_SIGNATURE_VALUE) {
|
|
|
|
splx(x);
|
2012-10-27 21:17:22 +04:00
|
|
|
aprint_error_dev(sc->sc_dev, "microcode signature bad: should be %04x, is %04x\n",
|
2008-04-08 16:07:25 +04:00
|
|
|
DAIC_SIGNATURE_VALUE,signature);
|
2001-01-05 15:49:52 +03:00
|
|
|
return EIO;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
splx(x);
|
2002-03-22 12:54:15 +03:00
|
|
|
tsleep(&sc->sc_port[i].du_state, 0, "daic protocol init", hz/25);
|
2001-01-05 15:49:52 +03:00
|
|
|
x = splnet();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* real check: send an invalid request and wait for an interrupt */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[i].du_state = DAIC_STATE_TESTING;
|
2001-01-05 15:49:52 +03:00
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RC+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_REQID+off, 0xff);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_REQ+off, 1);
|
|
|
|
splx(x);
|
2002-03-22 12:54:15 +03:00
|
|
|
tsleep(&sc->sc_port[i].du_state, 0, "daic irq test", 2*hz);
|
2001-01-05 15:49:52 +03:00
|
|
|
x = splnet();
|
2002-03-22 12:54:15 +03:00
|
|
|
if (sc->sc_port[i].du_state != DAIC_STATE_RUNNING) {
|
2001-01-05 15:49:52 +03:00
|
|
|
splx(x);
|
|
|
|
printf("%s: download interrupt test timeout\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev));
|
2001-01-05 15:49:52 +03:00
|
|
|
return EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* finish card configuration */
|
|
|
|
bus_space_write_4(sc->sc_iot, sc->sc_ioh, DAIC_SWID+off, sw_id);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_SET_CARD+off, sc->sc_cardtype);
|
|
|
|
splx(x);
|
|
|
|
|
|
|
|
/* assign global d-channel id for that port */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[i].du_global_dchan =
|
2001-01-05 15:49:52 +03:00
|
|
|
daic_assign(sc, i, DAIC_GLOBALID_DCHAN,
|
|
|
|
sizeof parm_global_assign, parm_global_assign);
|
|
|
|
|
|
|
|
/* send an INDICATE request to get incoming calls on this id */
|
|
|
|
x = splnet();
|
2002-03-22 12:54:15 +03:00
|
|
|
VOIDREQ(sc, i, DAIC_REQ_INDICATE, sc->sc_port[i].du_global_dchan);
|
2001-01-05 15:49:52 +03:00
|
|
|
splx(x);
|
2002-03-22 12:54:15 +03:00
|
|
|
tsleep(&sc->sc_port[i].du_request_res, 0, "daic request", 0);
|
2001-01-05 15:49:52 +03:00
|
|
|
x = splnet();
|
2002-03-22 12:54:15 +03:00
|
|
|
if (sc->sc_port[i].du_request_res != DAIC_RC_OK) {
|
2012-10-27 21:17:22 +04:00
|
|
|
aprint_error_dev(sc->sc_dev, "INDICATE request error (0x%02x): %s\n",
|
2008-04-08 16:07:25 +04:00
|
|
|
sc->sc_port[i].du_request_res,
|
2002-03-22 12:54:15 +03:00
|
|
|
err_codes[sc->sc_port[i].du_request_res & DAIC_RC_ERRMASK]);
|
2001-01-05 15:49:52 +03:00
|
|
|
splx(x);
|
|
|
|
return EIO;
|
|
|
|
}
|
|
|
|
splx(x);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Reset the card, download primary bootstrap, let it check the
|
|
|
|
* card and return the cardtype identified by the microcode
|
|
|
|
* or -1 if no known card is detected.
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static int
|
2009-03-14 18:35:58 +03:00
|
|
|
daic_reset(bus_space_tag_t bus, bus_space_handle_t io, int port, int *memsize)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
int i, off = port * DAIC_ISA_MEMSIZE;
|
2002-03-22 12:54:15 +03:00
|
|
|
int cardtype, mem, quiet = memsize == NULL; /* no output if we are only probing */
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* clear any pending interrupt */
|
|
|
|
bus_space_read_1(bus, io, DAIC_IRQ+off);
|
|
|
|
/* reset card */
|
|
|
|
bus_space_write_1(bus, io, DAIC_BOOT_SET_RESET+off, 0);
|
|
|
|
|
|
|
|
/* download primary bootstrap */
|
|
|
|
bus_space_set_region_1(bus, io, DAIC_BOOT_START+off, 0, DAIC_BOOT_CODE-DAIC_BOOT_START);
|
|
|
|
bus_space_write_region_1(bus, io, DAIC_BOOT_CODE+off, dnload, DAIC_BOOT_END-DAIC_BOOT_CODE+1);
|
|
|
|
if (bus_space_read_1(bus, io, DAIC_BOOT_CTRL+off)
|
|
|
|
|| bus_space_read_1(bus, io, DAIC_BOOT_EBIT+off)) {
|
|
|
|
if (!quiet) printf(": shared memory test failed!\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* let card perform memory test */
|
|
|
|
bus_space_write_1(bus, io, DAIC_BOOT_CTRL+off, DAIC_TEST_MEM);
|
|
|
|
/* and off we go... */
|
|
|
|
bus_space_write_1(bus, io, DAIC_BOOT_CLR_RESET+off, 0);
|
|
|
|
/* wait for response from bootstrap */
|
|
|
|
for (i = 0; i < 15000 && bus_space_read_1(bus, io, DAIC_BOOT_CTRL+off) != DAIC_TEST_RDY; i++)
|
|
|
|
DELAY(100);
|
|
|
|
if (i >= 15000) {
|
|
|
|
if (!quiet) printf(": on board processor test failed!\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (bus_space_read_1(bus, io, DAIC_BOOT_EBIT+off)) {
|
2010-07-27 12:07:36 +04:00
|
|
|
if (!quiet) printf(": on board memory test failed at %x\n",
|
|
|
|
bus_space_read_2(bus, io, DAIC_BOOT_ELOC+off));
|
2001-01-05 15:49:52 +03:00
|
|
|
return -1;
|
|
|
|
}
|
2005-02-27 03:26:58 +03:00
|
|
|
|
2001-01-05 15:49:52 +03:00
|
|
|
/* fetch info from primary bootstrap code */
|
|
|
|
cardtype = bus_space_read_1(bus, io, DAIC_BOOT_CARD+off);
|
2002-03-22 12:54:15 +03:00
|
|
|
mem = bus_space_read_1(bus, io, DAIC_BOOT_MSIZE+off) << 4;
|
|
|
|
if (memsize)
|
|
|
|
*memsize = mem;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
return cardtype;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Generic diagnostic interface - pass through the microcode data
|
|
|
|
* without knowing too much about it. This passes a lot work to
|
|
|
|
* userland, but hey, this is only a diagnostic tool...
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static int
|
2009-03-14 18:35:58 +03:00
|
|
|
daic_diagnostic(void *token, struct isdn_diagnostic_request *req)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_unit *du = token;
|
|
|
|
struct daic_softc *sc = du->du_sc;
|
|
|
|
int port = du->du_port;
|
2001-01-05 15:49:52 +03:00
|
|
|
int off = port * DAIC_ISA_MEMSIZE;
|
|
|
|
int rc, cnt;
|
|
|
|
int s, err = 0;
|
|
|
|
|
|
|
|
/* validate parameters */
|
2007-02-26 00:06:57 +03:00
|
|
|
if (req->cmd > DAIC_DIAG_MAXCMD) {
|
2012-10-27 21:17:22 +04:00
|
|
|
aprint_error_dev(sc->sc_dev, "daic_diagnostic: illegal cmd %d\n",
|
2008-04-08 16:07:25 +04:00
|
|
|
req->cmd);
|
2001-01-05 15:49:52 +03:00
|
|
|
return EIO;
|
|
|
|
}
|
|
|
|
if (req->out_param_len > (DAIC_DIAG_DATA_SIZE+1)) {
|
2012-10-27 21:17:22 +04:00
|
|
|
aprint_error_dev(sc->sc_dev, "daic_diagnostic: illegal out_param_len %d\n",
|
2008-04-08 16:07:25 +04:00
|
|
|
req->out_param_len);
|
2001-01-05 15:49:52 +03:00
|
|
|
return EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XXX - only for debug */
|
|
|
|
if (req->cmd == 0x05) {
|
|
|
|
/* pass through request from userland */
|
|
|
|
|
|
|
|
u_int8_t id;
|
|
|
|
static u_int8_t parms[] = {
|
|
|
|
IEI_CALLID, 0x01, 0x81,
|
|
|
|
IEI_CALLINGPN, 7, NUMBER_TYPEPLAN, '9', '8', '9', '0', '2', '0',
|
|
|
|
0x96, 0x01, 0x02, 0x01, 0x00,
|
|
|
|
0x00
|
|
|
|
};
|
|
|
|
|
|
|
|
/* create the d-channel task for this call */
|
2012-10-27 21:17:22 +04:00
|
|
|
printf("%s: assigning id for pass-through call\n", device_xname(sc->sc_dev));
|
2001-01-05 15:49:52 +03:00
|
|
|
id = daic_assign(sc, port, DAIC_GLOBALID_DCHAN, sizeof(parms), parms);
|
2012-10-27 21:17:22 +04:00
|
|
|
printf("%s: got id 0x%02x\n", device_xname(sc->sc_dev), id);
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
#ifdef DAIC_DEBUG
|
|
|
|
daic_dump_request(sc, port, DAIC_REQ_CALL, id, req->in_param_len, req->in_param);
|
|
|
|
#endif
|
|
|
|
daic_request(sc, port, DAIC_REQ_CALL, id, req->in_param_len, req->in_param);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* all these need an output parameter */
|
|
|
|
if (req->out_param == NULL)
|
|
|
|
return EIO;
|
|
|
|
|
|
|
|
/* check state and switch to DIAGNOSTIC */
|
|
|
|
s = splnet();
|
2002-03-22 12:54:15 +03:00
|
|
|
if (sc->sc_port[port].du_state != DAIC_STATE_RUNNING) {
|
2001-01-05 15:49:52 +03:00
|
|
|
splx(s);
|
|
|
|
return EWOULDBLOCK;
|
|
|
|
}
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_state = DAIC_STATE_DIAGNOSTIC;
|
2001-01-05 15:49:52 +03:00
|
|
|
splx(s);
|
|
|
|
|
|
|
|
/* set new request */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_REQ+off, req->cmd);
|
|
|
|
|
|
|
|
/* sorry, no interrupt on completition - have to poll */
|
|
|
|
for (cnt = 0; cnt < 3*hz; cnt++) {
|
|
|
|
if (bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_REQ+off) == 0
|
|
|
|
&& bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_RC+off) != 0)
|
|
|
|
break;
|
|
|
|
tsleep(sc, 0, "daic diagnostic", 1);
|
|
|
|
}
|
|
|
|
rc = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_RC+off);
|
|
|
|
if (rc == 0) {
|
|
|
|
/* stop request and return error */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_REQ+off, 0);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_RC+off, 0);
|
|
|
|
err = EIO;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
/* out param gets rc and all the data */
|
|
|
|
if (req->out_param_len >= 2) {
|
|
|
|
((u_int8_t*)(req->out_param))[0] = (u_int8_t)rc;
|
|
|
|
bus_space_read_region_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_DATA+off, ((u_int8_t*)req->out_param)+1, req->out_param_len-1);
|
|
|
|
}
|
|
|
|
/* acknowledge data */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_DIAG_RC+off, 0);
|
|
|
|
|
|
|
|
done: /* back to normal state */
|
|
|
|
s = splnet();
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_state = DAIC_STATE_RUNNING;
|
2001-01-05 15:49:52 +03:00
|
|
|
splx(s);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2002-03-22 12:54:15 +03:00
|
|
|
static void daic_stat(void *port, int channel, bchan_statistics_t *bsp)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct isdn_l4_bchannel_functions
|
|
|
|
daic_l4_driver = {
|
|
|
|
daic_bch_config,
|
|
|
|
daic_bch_tx_start,
|
|
|
|
daic_stat
|
|
|
|
};
|
|
|
|
|
2002-03-24 23:35:43 +03:00
|
|
|
static const struct isdn_l3_driver_functions
|
|
|
|
daic_l3_functions = {
|
|
|
|
daic_ret_linktab,
|
|
|
|
daic_set_link,
|
|
|
|
daic_connect_request,
|
|
|
|
daic_connect_response,
|
|
|
|
daic_disconnect_request,
|
|
|
|
daic_alert_request,
|
|
|
|
daic_download,
|
|
|
|
daic_diagnostic,
|
|
|
|
daic_mgmt_command
|
|
|
|
};
|
|
|
|
|
2001-01-05 15:49:52 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Register one port and attach it to the upper layers
|
|
|
|
*---------------------------------------------------------------------------*/
|
2002-03-24 23:35:43 +03:00
|
|
|
static void
|
2002-03-22 12:54:15 +03:00
|
|
|
daic_register_port(struct daic_softc *sc, int port)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
int chan;
|
2002-03-24 23:35:43 +03:00
|
|
|
char cardname[80], devname[80];
|
2002-03-29 23:29:53 +03:00
|
|
|
struct isdn_l3_driver * l3drv;
|
2002-03-22 12:54:15 +03:00
|
|
|
|
|
|
|
sc->sc_port[port].du_port = port;
|
|
|
|
sc->sc_port[port].du_sc = sc;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* make sure this hardware driver type is known to layer 4 */
|
2002-03-24 23:35:43 +03:00
|
|
|
if (sc->sc_cardtype == DAIC_TYPE_QUAD)
|
2004-04-22 04:17:10 +04:00
|
|
|
snprintf(devname, sizeof(devname), "%s port %d",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), port);
|
2002-03-24 23:35:43 +03:00
|
|
|
else
|
2012-10-27 21:17:22 +04:00
|
|
|
strlcpy(devname, device_xname(sc->sc_dev), sizeof(devname));
|
2004-04-22 04:17:10 +04:00
|
|
|
snprintf(cardname, sizeof(cardname), "EICON.Diehl %s",
|
|
|
|
cardtypename(sc->sc_cardtype));
|
2003-10-07 23:02:36 +04:00
|
|
|
l3drv = isdn_attach_isdnif(
|
2004-08-13 15:25:58 +04:00
|
|
|
devname, cardname, &sc->sc_port[port], &daic_l3_functions,
|
|
|
|
NBCH_BRI);
|
2002-03-29 23:29:53 +03:00
|
|
|
sc->sc_port[port].du_l3 = l3drv;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* initialize linktabs for this port */
|
|
|
|
for (chan = 0; chan < 2; chan++) {
|
|
|
|
isdn_link_t *lt = &sc->sc_con[port*2+chan].isdn_linktab;
|
2002-03-22 12:54:15 +03:00
|
|
|
lt->l1token = &sc->sc_port[port];
|
2001-01-05 15:49:52 +03:00
|
|
|
lt->channel = chan;
|
|
|
|
lt->tx_queue = &sc->sc_con[port*2+chan].tx_queue;
|
|
|
|
lt->rx_queue = &sc->sc_con[port*2+chan].rx_queue;
|
|
|
|
}
|
|
|
|
TAILQ_INIT(&sc->sc_outcalls[port]);
|
2002-03-29 23:29:53 +03:00
|
|
|
|
2003-10-07 23:02:36 +04:00
|
|
|
isdn_isdnif_ready(l3drv->isdnif);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
2005-02-27 03:26:58 +03:00
|
|
|
* return the address of daic drivers linktab
|
2001-01-05 15:49:52 +03:00
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static isdn_link_t *
|
2002-03-22 12:54:15 +03:00
|
|
|
daic_ret_linktab(void *token, int channel)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_unit *du = token;
|
|
|
|
struct daic_softc *sc = du->du_sc;
|
|
|
|
int port = du->du_port;
|
2001-01-05 15:49:52 +03:00
|
|
|
struct daic_connection *con = &sc->sc_con[port*2+channel];
|
|
|
|
|
|
|
|
return(&con->isdn_linktab);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* set the driver linktab in the b channel softc
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
2002-03-22 12:54:15 +03:00
|
|
|
daic_set_link(void *token, int channel, const struct isdn_l4_driver_functions *l4_driver, void *l4_inst)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_unit *du = token;
|
|
|
|
struct daic_softc *sc = du->du_sc;
|
|
|
|
int port = du->du_port;
|
2001-01-05 15:49:52 +03:00
|
|
|
struct daic_connection *con = &sc->sc_con[port*2+channel];
|
|
|
|
|
2002-03-22 12:54:15 +03:00
|
|
|
con->l4_driver = l4_driver;
|
|
|
|
con->l4_driver_softc = l4_inst;
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Send a request to the card.
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
daic_request(
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_softc *sc, /* ourself */
|
2001-01-05 15:49:52 +03:00
|
|
|
int port, /* and the port on this card */
|
|
|
|
u_int req, /* the request to send */
|
|
|
|
u_int id, /* id of communication task */
|
|
|
|
bus_size_t parmsize, /* size of parms including the terminating zero */
|
|
|
|
const u_int8_t *parms) /* pointer to parms to pass */
|
|
|
|
{
|
|
|
|
int off = port*DAIC_ISA_MEMSIZE;
|
|
|
|
|
|
|
|
/* spin while card is yet busy */
|
|
|
|
while (bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_REQ))
|
|
|
|
; /* unlikely to happen with this driver */
|
|
|
|
|
|
|
|
/* output parameters */
|
|
|
|
bus_space_write_region_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_XBUFFER+off, parms, parmsize);
|
|
|
|
|
|
|
|
/* output request and id */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_REQID+off, id);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_REQ+off, req);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Assign a unique instance id for some communication class
|
|
|
|
* on the card. Only one assign request may be running on a
|
|
|
|
* port at any time, handle this and return the instance id.
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static u_int
|
|
|
|
daic_assign(
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_softc *sc, /* our state and port no */
|
2001-01-05 15:49:52 +03:00
|
|
|
int port,
|
|
|
|
u_int classid, /* Diehl calls this "global instance id" */
|
2004-02-24 18:22:01 +03:00
|
|
|
bus_size_t parmsize, /* sizeof parameter arra */
|
2001-01-05 15:49:52 +03:00
|
|
|
const u_int8_t *parms) /* task instance parameters */
|
|
|
|
{
|
|
|
|
static char wchan[] = "daic assign";
|
|
|
|
u_int8_t id;
|
|
|
|
int x;
|
|
|
|
|
|
|
|
/* there only may be one assignment running concurrently */
|
|
|
|
x = splnet();
|
|
|
|
for (;;) {
|
2002-03-22 12:54:15 +03:00
|
|
|
if (!(sc->sc_port[port].du_assign & DAIC_ASSIGN_PENDING))
|
2001-01-05 15:49:52 +03:00
|
|
|
break; /* we got it! */
|
|
|
|
|
|
|
|
/* somebody else is assigning, record state and sleep */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_assign |= DAIC_ASSIGN_SLEEPING;
|
|
|
|
tsleep(&sc->sc_port[port].du_assign_res, 0, wchan, 0);
|
2001-01-05 15:49:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* put parameters and request to card */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_assign |= DAIC_ASSIGN_PENDING;
|
2001-01-05 15:49:52 +03:00
|
|
|
daic_request(sc, port, DAIC_REQ_ASSIGN, classid, parmsize, parms);
|
|
|
|
|
|
|
|
/* wait for completition of assignment by the card */
|
2002-03-22 12:54:15 +03:00
|
|
|
tsleep(&sc->sc_port[port].du_assign, 0, wchan, 0);
|
|
|
|
id = sc->sc_port[port].du_assign_res;
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* have we lost our global dchannel id in the meantime? */
|
2002-03-22 12:54:15 +03:00
|
|
|
if (sc->sc_port[port].du_assign & DAIC_ASSIGN_NOGLOBAL) {
|
2001-01-05 15:49:52 +03:00
|
|
|
/* start an assign request and let the result
|
|
|
|
be handled by the interrupt handler - we don't
|
|
|
|
have to wait for it here. As the assign lock
|
|
|
|
isn't freed, we don't wake up others... */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_assign &= ~DAIC_ASSIGN_NOGLOBAL;
|
|
|
|
sc->sc_port[port].du_assign |= DAIC_ASSIGN_PENDING|DAIC_ASSIGN_GLOBAL;
|
2005-02-27 03:26:58 +03:00
|
|
|
daic_request(sc, port, DAIC_REQ_ASSIGN, DAIC_GLOBALID_DCHAN,
|
2001-01-05 15:49:52 +03:00
|
|
|
sizeof parm_global_assign, parm_global_assign);
|
|
|
|
splx(x);
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XXX - review this, can't remember why I did it this complicated */
|
|
|
|
|
|
|
|
/* unlock and wakup others, if any */
|
2002-03-22 12:54:15 +03:00
|
|
|
if (sc->sc_port[port].du_assign & DAIC_ASSIGN_SLEEPING) {
|
|
|
|
sc->sc_port[port].du_assign = 0;
|
|
|
|
wakeup(&sc->sc_port[port].du_assign_res);
|
2001-01-05 15:49:52 +03:00
|
|
|
} else
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_assign = 0;
|
2001-01-05 15:49:52 +03:00
|
|
|
splx(x);
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DAIC_DEBUG
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Debug output of request parameters
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
2002-03-22 12:54:15 +03:00
|
|
|
daic_dump_request(struct daic_softc *sc, int port, u_int req, u_int id, bus_size_t parmsize, u_int8_t *parms)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
printf("%s: request 0x%02x to task id 0x%02x:",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), req, id);
|
2001-01-05 15:49:52 +03:00
|
|
|
for (i = 0; i < parmsize; i++) {
|
|
|
|
if (i % 16 == 0)
|
|
|
|
printf("\n%02x:", i);
|
|
|
|
printf(" %02x", parms[i]);
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Decode parameters of an INDICATE indication from the card
|
|
|
|
* and pass them to layer 4. Called from within an interrupt
|
|
|
|
* context.
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
2002-03-22 12:54:15 +03:00
|
|
|
daic_indicate_ind(struct daic_softc *sc, int port)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
int offset = port*DAIC_ISA_MEMSIZE;
|
|
|
|
int i;
|
|
|
|
u_int8_t ie, ielen;
|
|
|
|
call_desc_t *cd;
|
|
|
|
|
|
|
|
/* get and init new calldescriptor */
|
|
|
|
cd = reserve_cd(); /* cdid filled in */
|
|
|
|
cd->bprot = BPROT_NONE;
|
|
|
|
cd->cause_in = 0;
|
2005-02-27 03:26:58 +03:00
|
|
|
cd->cause_out = 0;
|
2001-01-05 15:49:52 +03:00
|
|
|
cd->dst_telno[0] = '\0';
|
|
|
|
cd->src_telno[0] = '\0';
|
|
|
|
cd->channelid = CHAN_NO;
|
|
|
|
cd->channelexcl = 0;
|
|
|
|
cd->cr = -1;
|
|
|
|
cd->crflag = CRF_DEST;
|
|
|
|
cd->ilt = NULL; /* reset link tab ptrs */
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
for (;;) {
|
|
|
|
ie = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RBUFFER+offset+i);
|
|
|
|
if (!ie) break;
|
|
|
|
i++;
|
|
|
|
if (ie & 0x80) continue;
|
|
|
|
ielen = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RBUFFER+offset+i);
|
|
|
|
i++;
|
|
|
|
switch (ie) {
|
|
|
|
case IEI_BEARERCAP:
|
|
|
|
ie = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RBUFFER+offset+i);
|
|
|
|
if (ie == 0x80 || ie == 0x89 || ie == 0x90)
|
|
|
|
cd->bprot = BPROT_NONE;
|
|
|
|
else if (ie == 0x88)
|
|
|
|
cd->bprot = BPROT_RHDLC;
|
|
|
|
break;
|
|
|
|
case IEI_CALLINGPN:
|
|
|
|
{
|
|
|
|
int off;
|
|
|
|
ie = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RBUFFER+offset+i);
|
|
|
|
if (ie & 0x80)
|
|
|
|
off = 1;
|
|
|
|
else
|
|
|
|
off = 2;
|
2005-02-27 03:26:58 +03:00
|
|
|
bus_space_read_region_1(sc->sc_iot, sc->sc_ioh,
|
2001-01-05 15:49:52 +03:00
|
|
|
DAIC_COM_RBUFFER+offset+i+off, cd->src_telno,
|
|
|
|
ielen - off);
|
|
|
|
cd->src_telno[ielen-off+1] = '\0';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IEI_CALLEDPN:
|
|
|
|
bus_space_read_region_1(sc->sc_iot, sc->sc_ioh,
|
2005-02-27 03:26:58 +03:00
|
|
|
DAIC_COM_RBUFFER+offset+i+1,
|
2001-01-05 15:49:52 +03:00
|
|
|
cd->dst_telno, ielen-1);
|
|
|
|
cd->dst_telno[ielen] = '\0';
|
|
|
|
break;
|
|
|
|
case IEI_CHANNELID:
|
|
|
|
ie = bus_space_read_1(sc->sc_iot, sc->sc_ioh, DAIC_COM_RBUFFER+offset+i);
|
|
|
|
if ((ie & 0xf4) != 0x80)
|
|
|
|
cd->channelid = CHAN_NO;
|
|
|
|
else {
|
|
|
|
switch(ie & 0x03) {
|
|
|
|
case IE_CHAN_ID_NO: cd->channelid = CHAN_NO; break;
|
|
|
|
case IE_CHAN_ID_B1: cd->channelid = CHAN_B1; break;
|
|
|
|
case IE_CHAN_ID_B2: cd->channelid = CHAN_B2; break;
|
|
|
|
case IE_CHAN_ID_ANY: cd->channelid = CHAN_ANY; break;
|
|
|
|
}
|
|
|
|
cd->channelexcl = (ie & 0x08) >> 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
i += ielen;
|
|
|
|
}
|
|
|
|
cd->event = EV_SETUP;
|
2002-03-22 12:54:15 +03:00
|
|
|
/* ctrl_desc[cd->controller].bch_state[cd->channelid] = BCH_ST_RSVD; */
|
2001-01-05 15:49:52 +03:00
|
|
|
|
|
|
|
/* record the dchannel id for this call and the call descriptor */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_con[port*2+cd->channelid].dchan_inst = sc->sc_port[port].du_global_dchan;
|
2001-01-05 15:49:52 +03:00
|
|
|
sc->sc_con[port*2+cd->channelid].cdid = cd->cdid;
|
|
|
|
|
|
|
|
/* this task is busy now, we need a new global dchan id */
|
2002-03-22 12:54:15 +03:00
|
|
|
if (sc->sc_port[port].du_assign & DAIC_ASSIGN_PENDING) {
|
2001-01-05 15:49:52 +03:00
|
|
|
/* argh - can't assign right now */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_assign |= DAIC_ASSIGN_NOGLOBAL;
|
2001-01-05 15:49:52 +03:00
|
|
|
} else {
|
|
|
|
/* yeah - can request the assign right away, but let the
|
|
|
|
interrupt handler autohandle the result */
|
2002-03-22 12:54:15 +03:00
|
|
|
sc->sc_port[port].du_assign |= DAIC_ASSIGN_PENDING|DAIC_ASSIGN_GLOBAL;
|
2001-01-05 15:49:52 +03:00
|
|
|
daic_request(sc, port, DAIC_REQ_ASSIGN, DAIC_GLOBALID_DCHAN,
|
|
|
|
sizeof parm_global_assign, parm_global_assign);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cd->bprot == BPROT_NONE)
|
|
|
|
printf("\nincoming voice call from \"%s\" to \"%s\"\n",
|
|
|
|
cd->src_telno, cd->dst_telno);
|
|
|
|
else
|
|
|
|
printf("\nincoming data call from \"%s\" to \"%s\"\n",
|
|
|
|
cd->src_telno, cd->dst_telno);
|
|
|
|
|
|
|
|
/* hand up call to layer 4 */
|
|
|
|
i4b_l4_connect_ind(cd);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* Layer 4 request a call setup
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
2002-03-30 14:15:41 +03:00
|
|
|
daic_connect_request(struct call_desc *cd)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
u_int8_t id, cpn[TELNO_MAX+4], parms[TELNO_MAX+16], *p;
|
2002-03-22 12:54:15 +03:00
|
|
|
struct daic_unit *du = cd->ilt->l1token;
|
|
|
|
struct daic_softc *sc = du->du_sc;
|
|
|
|
int port = du->du_port;
|
2001-01-05 15:49:52 +03:00
|
|
|
int x, len;
|
|
|
|
struct outcallentry *assoc;
|
|
|
|
|
|
|
|
/* to associate the cdid with the communication task
|
|
|
|
we are going to create for this outgoing call,
|
|
|
|
we maintain a queue of pending outgoing calls.
|
|
|
|
As soon as a SETUP response is received, we move
|
|
|
|
the association to the allocated b-channel. */
|
|
|
|
|
|
|
|
/* configure d-channel task parameters */
|
|
|
|
p = parms;
|
|
|
|
*p++ = IEI_CALLID; *p++ = 0x01;
|
|
|
|
if (cd->bprot == BPROT_NONE) {
|
|
|
|
*p++ = 0x82;
|
|
|
|
} else if (cd->bprot == BPROT_RHDLC) {
|
|
|
|
*p++ = 0x85;
|
|
|
|
} else {
|
|
|
|
printf("%s: daic_connect_request for unknown bchan protocol 0x%x\n",
|
2012-10-27 21:17:22 +04:00
|
|
|
device_xname(sc->sc_dev), cd->bprot);
|
2001-01-05 15:49:52 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (cd->src_telno[0]) {
|
|
|
|
*p++ = IEI_CALLINGPN;
|
|
|
|
*p++ = strlen(cd->src_telno)+1;
|
|
|
|
*p++ = NUMBER_TYPEPLAN;
|
|
|
|
strcpy(p, cd->src_telno);
|
|
|
|
p += strlen(p);
|
|
|
|
}
|
|
|
|
if (cd->channelid == CHAN_B1 || cd->channelid == CHAN_B2) {
|
|
|
|
*p++ = IEI_CHANNELID;
|
|
|
|
*p++ = 0x01;
|
|
|
|
*p++ = 0x81 + cd->channelid;
|
|
|
|
}
|
|
|
|
if (cd->bprot == BPROT_NONE) {
|
|
|
|
*p++ = 0x96; /* shift6 */
|
|
|
|
*p++ = 0x01; /* SIN */
|
|
|
|
*p++ = 0x02; /* len */
|
|
|
|
*p++ = 0x01; /* Telephony */
|
|
|
|
*p++ = 0x00; /* add.info */
|
|
|
|
}
|
|
|
|
*p++ = 0;
|
|
|
|
|
|
|
|
/* create the d-channel task for this call */
|
|
|
|
id = daic_assign(sc, port, DAIC_GLOBALID_DCHAN, p - parms, parms);
|
|
|
|
|
|
|
|
/* map it to the call descriptor id */
|
2003-02-05 02:26:36 +03:00
|
|
|
assoc = malloc(sizeof(struct outcallentry), M_DEVBUF, 0);
|
2002-03-30 14:15:41 +03:00
|
|
|
assoc->cdid = cd->cdid;
|
2001-01-05 15:49:52 +03:00
|
|
|
assoc->dchan_id = id;
|
|
|
|
x = splnet();
|
|
|
|
TAILQ_INSERT_TAIL(&sc->sc_outcalls[port], assoc, queue);
|
|
|
|
|
|
|
|
/* send a call request */
|
|
|
|
len = strlen(cd->dst_telno);
|
|
|
|
cpn[0] = IEI_CALLEDPN;
|
|
|
|
cpn[1] = len+1;
|
|
|
|
cpn[2] = NUMBER_TYPEPLAN;
|
|
|
|
strcpy(cpn+3, cd->dst_telno);
|
|
|
|
#ifdef DAIC_DEBUG
|
|
|
|
daic_dump_request(sc, port, DAIC_REQ_CALL, id, len+4, cpn);
|
|
|
|
#endif
|
|
|
|
daic_request(sc, port, DAIC_REQ_CALL, id, len+4, cpn);
|
|
|
|
splx(x);
|
|
|
|
tsleep(assoc, 0, "daic call", 0);
|
|
|
|
if (assoc->rc != DAIC_RC_OK) {
|
2012-10-27 21:17:22 +04:00
|
|
|
aprint_error_dev(sc->sc_dev, "call request failed, error 0x%02x: %s\n",
|
2008-04-08 16:07:25 +04:00
|
|
|
assoc->rc & DAIC_RC_ERRMASK,
|
2001-01-05 15:49:52 +03:00
|
|
|
err_codes[assoc->rc & DAIC_RC_ERRMASK]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* TODO:
|
|
|
|
*---------------------------------------------------------------------------*/
|
2002-03-30 14:15:41 +03:00
|
|
|
static void daic_connect_response(struct call_desc *cd, int response, int cause)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* TODO:
|
|
|
|
*---------------------------------------------------------------------------*/
|
2002-03-30 14:15:41 +03:00
|
|
|
static void daic_disconnect_request(struct call_desc *cd, int cause)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* TODO:
|
|
|
|
*---------------------------------------------------------------------------*/
|
2002-03-22 12:54:15 +03:00
|
|
|
static void daic_bch_config(void *token, int channel, int bprot, int updown)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
printf("daic: bch_config\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* TODO:
|
|
|
|
*---------------------------------------------------------------------------*/
|
2002-03-22 12:54:15 +03:00
|
|
|
static void daic_bch_tx_start(void *token, int channel)
|
2001-01-05 15:49:52 +03:00
|
|
|
{
|
|
|
|
printf("daic: bch_tx_start\n");
|
|
|
|
}
|
|
|
|
|
2002-03-22 12:54:15 +03:00
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* TODO:
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
2002-03-30 14:15:41 +03:00
|
|
|
daic_mgmt_command(struct isdn_l3_driver *drv, int cmd, void *parm)
|
2002-03-22 12:54:15 +03:00
|
|
|
{
|
|
|
|
}
|
2002-03-24 23:35:43 +03:00
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* TODO:
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
2002-03-30 14:15:41 +03:00
|
|
|
daic_alert_request(struct call_desc *cd)
|
2002-03-24 23:35:43 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|