summaryrefslogblamecommitdiffstats
path: root/loop.c
blob: cbd82e5e2ccbe8013f278c197683e6967ffb674d (plain) (tree)
























































































































































                                                                                                                                          
/*****************************************************************************\
**                                                                           **
** LCR                                                                       **
**                                                                           **
**---------------------------------------------------------------------------**
** Copyright: Andreas Eversberg                                              **
**                                                                           **
** loopback interface functions                                              **
**                                                                           **
\*****************************************************************************/ 

#include "main.h"

struct mISDNloop mISDNloop = { -1, 0 };

void mISDNloop_close(void)
{
	if (mISDNloop.sock > -1)
		close(mISDNloop.sock);
	mISDNloop.sock = -1;
}

int mISDNloop_open()
{
	int ret;
	int cnt;
	unsigned long on = 1;
	struct sockaddr_mISDN addr;
	struct mISDN_devinfo devinfo;
	int pri, bri;

	/* already open */
	if (mISDNloop.sock > -1)
		return 0;

	PDEBUG(DEBUG_PORT, "Open external interface of loopback.\n");

	/* check port counts */
	ret = ioctl(mISDNsocket, IMGETCOUNT, &cnt);
	if (ret < 0) {
		fprintf(stderr, "Cannot get number of mISDN devices. (ioctl IMGETCOUNT failed ret=%d)\n", ret);
		return(ret);
	}

	if (cnt <= 0) {
		PERROR_RUNTIME("Found no card. Please be sure to load card drivers.\n");
		return -EIO;
	}
	mISDNloop.port = mISDN_getportbyname(mISDNsocket, cnt, options.loopback_ext);
	if (mISDNloop.port < 0) {
		PERROR_RUNTIME("Port name '%s' not found, did you load loopback interface?.\n", options.loopback_ext);
		return mISDNloop.port;
	}
	/* get protocol */
	bri = pri = 0;
	devinfo.id = mISDNloop.port;
	ret = ioctl(mISDNsocket, IMGETDEVINFO, &devinfo);
	if (ret < 0) {
		PERROR_RUNTIME("Cannot get device information for port %d. (ioctl IMGETDEVINFO failed ret=%d)\n", mISDNloop.port, ret);
		return ret;
	}
	if (devinfo.Dprotocols & (1 << ISDN_P_TE_S0)) {
		bri = 1;
	}
	if (devinfo.Dprotocols & (1 << ISDN_P_TE_E1)) {
		pri = 1;
	}
	if (!bri && !pri) {
		PERROR_RUNTIME("loop port %d does not support TE PRI or TE BRI.\n", mISDNloop.port);
	}
	/* open socket */
	if ((mISDNloop.sock = socket(PF_ISDN, SOCK_DGRAM, (pri)?ISDN_P_TE_E1:ISDN_P_TE_S0)) < 0) {
		PERROR_RUNTIME("loop port %d failed to open socket.\n", mISDNloop.port);
		mISDNloop_close();
		return mISDNloop.sock;
	}
	/* set nonblocking io */
	if ((ret = ioctl(mISDNloop.sock, FIONBIO, &on)) < 0) {
		PERROR_RUNTIME("loop port %d failed to set socket into nonblocking io.\n", mISDNloop.port);
		mISDNloop_close();
		return ret;
	}
	/* bind socket to dchannel */
	memset(&addr, 0, sizeof(addr));
	addr.family = AF_ISDN;
	addr.dev = mISDNloop.port;
	addr.channel = 0;
	if ((ret = bind(mISDNloop.sock, (struct sockaddr *)&addr, sizeof(addr))) < 0) {
		PERROR_RUNTIME("loop port %d failed to bind socket. (name = %s errno=%d)\n", mISDNloop.port, options.loopback_ext, errno);
		mISDNloop_close();
		return (ret);
	}

	return 0;
}

int loop_hunt_bchannel(class PmISDN *port, struct mISDNport *mISDNport)
{
	int channel;
	int i;
	char map[mISDNport->b_num];
	struct interface *interface;
	struct interface_port *ifport;

	chan_trace_header(mISDNport, port, "CHANNEL SELECTION (setup)", DIRECTION_NONE);
	add_trace("channel", "reserved", "%d", mISDNport->b_reserved);
	if (mISDNport->b_reserved >= mISDNport->b_num) { // of out chan..
		add_trace("conclusion", NULL, "all channels are reserved");
		end_trace();
		return(-34); // no channel
	}

	/* map all used ports of shared loopback interface */
	memset(map, 0, sizeof(map));
	interface = interface_first;
	while(interface) {
		ifport = interface->ifport;
		while(ifport) {
			if (!strcmp(ifport->portname, options.loopback_lcr)) {
				i = 0;
				while(i < mISDNport->b_num) {
					if (mISDNport->b_port[i])
						map[i] = 1;
					i++;
				}
			}
			ifport = ifport->next;
		}
		interface = interface->next;
	}

	/* find channel */
	i = 0;
	channel = 0;
	while(i < mISDNport->b_num) {
		if (!map[i]) {
			channel = i+1+(i>=15);
			break;
		}
		i++;
	}
	if (!channel) {
		add_trace("conclusion", NULL, "no channel available");
		end_trace();
		return(-6); // channel unacceptable
	}
	add_trace("conclusion", NULL, "channel available");
	add_trace("connect", "channel", "%d", channel);
	end_trace();
	return(channel);
}