2001-01-06 16:00:10 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1997, 1999 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 daemon - main program entry
|
|
|
|
* -------------------------------
|
|
|
|
*
|
2003-10-06 13:43:27 +04:00
|
|
|
* $Id: main.c,v 1.8 2003/10/06 09:43:27 itojun Exp $
|
2001-01-06 16:00:10 +03:00
|
|
|
*
|
|
|
|
* $FreeBSD$
|
|
|
|
*
|
2001-01-08 10:18:54 +03:00
|
|
|
* last edit-date: [Mon Jan 8 07:57:26 2001]
|
2001-01-06 16:00:10 +03:00
|
|
|
*
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#include <locale.h>
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
#include "monitor.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define MAIN
|
|
|
|
#include "isdnd.h"
|
|
|
|
#undef MAIN
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
|
|
|
|
#ifdef I4B_NOTCPIP_MONITOR
|
|
|
|
/* monitor via local socket */
|
|
|
|
static void mloop(int sockfd);
|
|
|
|
#else /* I4B_NOTCPIP_MONITOR */
|
|
|
|
/* monitor via local and tcp/ip socket */
|
|
|
|
static void mloop(int localsock, int remotesock);
|
|
|
|
#endif /* I4B_NOTCPIP_MONITOR */
|
|
|
|
|
|
|
|
#else /* I4B_EXTERNAL_MONITOR */
|
|
|
|
/* no monitoring at all */
|
|
|
|
static void mloop();
|
|
|
|
#endif /* I4B_EXTERNAL_MONITOR */
|
|
|
|
|
|
|
|
#ifdef USE_CURSES
|
|
|
|
static void kbdrdhdl(void);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void isdnrdhdl(void);
|
|
|
|
static void usage(void);
|
|
|
|
|
|
|
|
#define MSG_BUF_SIZ 1024 /* message buffer size */
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* usage display and exit
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
usage(void)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
fprintf(stderr, "isdnd - i4b ISDN manager daemon, version %02d.%02d.%d, %s %s\n", VERSION, REL, STEP, __DATE__, __TIME__);
|
|
|
|
#ifdef DEBUG
|
|
|
|
fprintf(stderr, " usage: isdnd [-c file] [-d level] [-F] [-f [-r dev] [-t termtype]]\n");
|
|
|
|
#else
|
|
|
|
fprintf(stderr, " usage: isdnd [-c file] [-F] [-f [-r dev] [-t termtype]]\n");
|
|
|
|
#endif
|
|
|
|
fprintf(stderr, " [-l] [-L file] [-m] [-s facility] [-u time]\n");
|
|
|
|
fprintf(stderr, " -c <filename> configuration file name (def: %s)\n", CONFIG_FILE_DEF);
|
|
|
|
#ifdef DEBUG
|
|
|
|
fprintf(stderr, " -d <level> set debug flag bits:\n");
|
|
|
|
fprintf(stderr, " general = 0x%04x, rates = 0x%04x, timing = 0x%04x\n", DL_MSG, DL_RATES, DL_TIME);
|
|
|
|
fprintf(stderr, " state = 0x%04x, retry = 0x%04x, dial = 0x%04x\n", DL_STATE, DL_RCVRY, DL_DIAL);
|
|
|
|
fprintf(stderr, " process = 0x%04x, kernio = 0x%04x ctrlstat = 0x%04x\n", DL_PROC, DL_DRVR, DL_CNST);
|
|
|
|
fprintf(stderr, " rc-file = 0x%04x, budget = 0x%04x\n", DL_RCCF, DL_BDGT);
|
|
|
|
fprintf(stderr, " -dn no debug output on fullscreen display\n");
|
|
|
|
#endif
|
|
|
|
fprintf(stderr, " -f fullscreen status display\n");
|
|
|
|
fprintf(stderr, " -F do not become a daemon process\n");
|
|
|
|
fprintf(stderr, " -l use a logfile instead of syslog\n");
|
|
|
|
fprintf(stderr, " -L <file> use file instead of %s for logging\n", LOG_FILE_DEF);
|
|
|
|
fprintf(stderr, " -P pretty print real config to stdout and exit\n");
|
|
|
|
fprintf(stderr, " -r <device> redirect output to other device (for -f)\n");
|
|
|
|
fprintf(stderr, " -s <facility> use facility instead of %d for syslog logging\n", LOG_LOCAL0 >> 3);
|
|
|
|
fprintf(stderr, " -t <termtype> terminal type of redirected screen (for -f)\n");
|
|
|
|
fprintf(stderr, " -u <time> length of a charging unit in seconds\n");
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
fprintf(stderr, " -m inhibit network/local monitoring (protocol %02d.%02d)\n", MPROT_VERSION, MPROT_REL);
|
|
|
|
#endif
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* program entry
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
msg_vr_req_t mvr;
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
int sockfd = -1; /* local monitor socket */
|
|
|
|
#ifndef I4B_NOTCPIP_MONITOR
|
|
|
|
int remotesockfd = -1; /* tcp/ip monitor socket */
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
setlocale (LC_ALL, "");
|
|
|
|
|
|
|
|
while ((i = getopt(argc, argv, "mc:d:fFlL:Pr:s:t:u:")) != -1)
|
|
|
|
{
|
|
|
|
switch (i)
|
|
|
|
{
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'm':
|
|
|
|
inhibit_monitor = 1;
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
#endif
|
2003-10-06 13:43:27 +04:00
|
|
|
|
|
|
|
case 'c':
|
|
|
|
configfile = optarg;
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'd':
|
|
|
|
if (*optarg == 'n')
|
|
|
|
debug_noscreen = 1;
|
|
|
|
else if ((sscanf(optarg, "%i", &debug_flags)) == 1)
|
|
|
|
do_debug = 1;
|
|
|
|
else
|
|
|
|
usage();
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
#endif
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'f':
|
|
|
|
do_fullscreen = 1;
|
|
|
|
do_fork = 0;
|
2001-01-06 16:00:10 +03:00
|
|
|
#ifndef USE_CURSES
|
2003-10-06 13:43:27 +04:00
|
|
|
fprintf(stderr, "Sorry, no fullscreen mode available - daemon compiled without USE_CURSES\n");
|
|
|
|
exit(1);
|
2001-01-06 16:00:10 +03:00
|
|
|
#endif
|
2003-10-06 13:43:27 +04:00
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'F':
|
|
|
|
do_fork = 0;
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'l':
|
|
|
|
uselogfile = 1;
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'L':
|
|
|
|
strlcpy(logfile, optarg, sizeof(logfile));
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'P':
|
|
|
|
do_print = 1;
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'r':
|
|
|
|
rdev = optarg;
|
|
|
|
do_rdev = 1;
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 's':
|
|
|
|
if (isdigit(*optarg))
|
|
|
|
{
|
|
|
|
int facility;
|
|
|
|
logfacility = strtoul(optarg, NULL, 10);
|
|
|
|
facility = logfacility << 3;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((facility < LOG_KERN) ||
|
|
|
|
(facility > LOG_FTP && facility < LOG_LOCAL0) ||
|
|
|
|
(facility > LOG_LOCAL7))
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
fprintf(stderr, "Error, option -s has invalid logging facility %d", logfacility);
|
2001-01-06 16:00:10 +03:00
|
|
|
usage();
|
|
|
|
}
|
2003-10-06 13:43:27 +04:00
|
|
|
logfacility = facility;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error: option -s requires a numeric argument!\n");
|
|
|
|
usage();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 't':
|
|
|
|
ttype = optarg;
|
|
|
|
do_ttytype = 1;
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
case 'u':
|
|
|
|
if (isdigit(*optarg))
|
|
|
|
{
|
|
|
|
unit_length = strtoul(optarg, NULL, 10);
|
|
|
|
if (unit_length < ULSRC_CMDLMIN)
|
|
|
|
unit_length = ULSRC_CMDLMIN;
|
|
|
|
else if (unit_length > ULSRC_CMDLMAX)
|
|
|
|
unit_length = ULSRC_CMDLMAX;
|
|
|
|
got_unitlen = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error: option -T requires a numeric argument!\n");
|
2001-01-06 16:00:10 +03:00
|
|
|
usage();
|
2003-10-06 13:43:27 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '?':
|
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef DEBUG
|
2003-10-06 13:43:27 +04:00
|
|
|
if (!do_debug)
|
2001-01-06 16:00:10 +03:00
|
|
|
debug_noscreen = 0;
|
|
|
|
#endif
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (!do_print)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
umask(UMASK); /* set our umask ... */
|
|
|
|
|
|
|
|
init_log(); /* initialize the logging subsystem */
|
|
|
|
}
|
|
|
|
|
|
|
|
check_pid(); /* check if we are already running */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (!do_print)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_fork || (do_fullscreen && do_rdev)) /* daemon mode ? */
|
2001-01-06 16:00:10 +03:00
|
|
|
daemonize();
|
|
|
|
|
|
|
|
write_pid(); /* write our pid to file */
|
|
|
|
|
|
|
|
/* set signal handler(s) */
|
|
|
|
|
|
|
|
signal(SIGCHLD, sigchild_handler); /* process handling */
|
|
|
|
signal(SIGHUP, rereadconfig); /* reread configuration */
|
|
|
|
signal(SIGUSR1, reopenfiles); /* reopen acct/log files*/
|
|
|
|
signal(SIGPIPE, SIG_IGN); /* handled manually */
|
|
|
|
signal(SIGINT, do_exit); /* clean up on SIGINT */
|
|
|
|
signal(SIGTERM, do_exit); /* clean up on SIGTERM */
|
|
|
|
signal(SIGQUIT, do_exit); /* clean up on SIGQUIT */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* open isdn device */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((isdnfd = open(I4BDEVICE, O_RDWR)) < 0)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "main: cannot open %s: %s", I4BDEVICE, strerror(errno));
|
2001-01-06 16:00:10 +03:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check kernel and userland have same version/release numbers */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((ioctl(isdnfd, I4B_VR_REQ, &mvr)) < 0)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "main: ioctl I4B_VR_REQ failed: %s", strerror(errno));
|
2001-01-06 16:00:10 +03:00
|
|
|
do_exit(1);
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (mvr.version != VERSION)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "main: version mismatch, kernel %d, daemon %d", mvr.version, VERSION);
|
2001-01-06 16:00:10 +03:00
|
|
|
do_exit(1);
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (mvr.release != REL)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "main: release mismatch, kernel %d, daemon %d", mvr.release, REL);
|
2001-01-06 16:00:10 +03:00
|
|
|
do_exit(1);
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (mvr.step != STEP)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "main: step mismatch, kernel %d, daemon %d", mvr.step, STEP);
|
2001-01-06 16:00:10 +03:00
|
|
|
do_exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* init controller state array */
|
|
|
|
|
|
|
|
init_controller();
|
|
|
|
|
|
|
|
/* read runtime configuration file and configure ourselves */
|
|
|
|
|
|
|
|
configure(configfile, 0);
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (config_error_flag)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "there were %d error(s) in the configuration file, terminating!", config_error_flag);
|
2001-01-06 16:00:10 +03:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set controller ISDN protocol */
|
|
|
|
|
|
|
|
init_controller_protocol();
|
|
|
|
|
|
|
|
/* init active controllers, if any */
|
|
|
|
|
|
|
|
signal(SIGCHLD, SIG_IGN); /*XXX*/
|
|
|
|
|
|
|
|
init_active_controller();
|
|
|
|
|
|
|
|
signal(SIGCHLD, sigchild_handler); /*XXX*/
|
|
|
|
|
|
|
|
/* handle the rates stuff */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((i = readrates(ratesfile)) == ERROR)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if (rate_error != NULL)
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "%s", rate_error);
|
2001-01-06 16:00:10 +03:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (i == GOOD)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
got_rate = 1; /* flag, ratesfile read and ok */
|
2002-12-06 18:00:14 +03:00
|
|
|
DBGL(DL_RCCF, (logit(LL_DBG, "ratesfile %s read successfully", ratesfile)));
|
2001-01-06 16:00:10 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if (rate_error != NULL)
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_WRN, "%s", rate_error);
|
2001-01-06 16:00:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* if writing accounting info, open file, set unbuffered */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (useacctfile)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((acctfp = fopen(acctfile, "a")) == NULL)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "ERROR, can't open acctfile %s for writing, terminating!", acctfile);
|
2001-01-06 16:00:10 +03:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
setvbuf(acctfp, (char *)NULL, _IONBF, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* initialize alias processing */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (aliasing)
|
2001-01-06 16:00:10 +03:00
|
|
|
init_alias(aliasfile);
|
|
|
|
|
|
|
|
/* init holidays */
|
|
|
|
|
|
|
|
init_holidays(holidayfile);
|
|
|
|
|
|
|
|
/* init remote monitoring */
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_monitor)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
monitor_init();
|
|
|
|
sockfd = monitor_create_local_socket();
|
|
|
|
#ifndef I4B_NOTCPIP_MONITOR
|
|
|
|
remotesockfd = monitor_create_remote_socket(monitorport);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* in case fullscreendisplay, initialize */
|
|
|
|
|
|
|
|
#ifdef USE_CURSES
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_fullscreen)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
init_screen();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* init realtime priority */
|
|
|
|
|
|
|
|
#ifdef USE_RTPRIO
|
2003-10-06 13:43:27 +04:00
|
|
|
if (rt_prio != RTPRIO_NOTUSED)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
struct rtprio rtp;
|
|
|
|
|
|
|
|
rtp.type = RTP_PRIO_REALTIME;
|
|
|
|
rtp.prio = rt_prio;
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((rtprio(RTP_SET, getpid(), &rtp)) == -1)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "rtprio failed: %s", strerror(errno));
|
2001-01-06 16:00:10 +03:00
|
|
|
do_exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
starttime = time(NULL); /* get starttime */
|
|
|
|
|
|
|
|
srandom(580403); /* init random number gen */
|
|
|
|
|
|
|
|
mloop( /* enter loop of no return .. */
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
sockfd
|
|
|
|
#ifndef I4B_NOTCPIP_MONITOR
|
|
|
|
, remotesockfd
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
do_exit(0);
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* program exit
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
do_exit(int exitval)
|
|
|
|
{
|
|
|
|
close_allactive();
|
|
|
|
|
|
|
|
unlink(PIDFILE);
|
|
|
|
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_DMN, "daemon terminating, exitval = %d", exitval);
|
2001-01-06 16:00:10 +03:00
|
|
|
|
|
|
|
#ifdef USE_CURSES
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_fullscreen && curses_ready)
|
2001-01-06 16:00:10 +03:00
|
|
|
endwin();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
monitor_exit();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
exit(exitval);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* program exit
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
error_exit(int exitval, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
close_allactive();
|
|
|
|
|
|
|
|
unlink(PIDFILE);
|
|
|
|
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_DMN, "fatal error, daemon terminating, exitval = %d", exitval);
|
2001-01-06 16:00:10 +03:00
|
|
|
|
|
|
|
#ifdef USE_CURSES
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_fullscreen && curses_ready)
|
2001-01-06 16:00:10 +03:00
|
|
|
endwin();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
monitor_exit();
|
|
|
|
#endif
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (mailto[0] && mailer[0])
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
|
|
|
|
#define EXITBL 2048
|
|
|
|
|
|
|
|
char ebuffer[EXITBL];
|
|
|
|
char sbuffer[EXITBL];
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vsnprintf(ebuffer, EXITBL-1, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
signal(SIGCHLD, SIG_IGN); /* remove handler */
|
|
|
|
|
|
|
|
snprintf(sbuffer, sizeof(sbuffer), "%s%s%s%s%s%s%s%s",
|
|
|
|
"cat << ENDOFDATA | ",
|
|
|
|
mailer,
|
|
|
|
" -s \"i4b isdnd: fatal error, terminating\" ",
|
|
|
|
mailto,
|
|
|
|
"\nThe isdnd terminated because of a fatal error:\n\n",
|
|
|
|
ebuffer,
|
|
|
|
"\n\nYours sincerely,\n the isdnd\n",
|
|
|
|
"\nENDOFDATA\n");
|
|
|
|
system(sbuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
exit(exitval);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* main loop
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
mloop(
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
|
|
|
int localmonitor
|
|
|
|
#ifndef I4B_NOTCPIP_MONITOR
|
|
|
|
, int remotemonitor
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
{
|
|
|
|
fd_set set;
|
|
|
|
struct timeval timeout;
|
|
|
|
int ret;
|
|
|
|
int high_selfd;
|
|
|
|
|
|
|
|
/* go into loop */
|
|
|
|
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_DMN, "i4b isdn daemon started (pid = %d)", getpid());
|
2001-01-06 16:00:10 +03:00
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
for (;;)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
FD_ZERO(&set);
|
|
|
|
|
|
|
|
#ifdef USE_CURSES
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_fullscreen)
|
2002-09-20 19:23:05 +04:00
|
|
|
FD_SET(STDIN_FILENO, &set);
|
2001-01-06 16:00:10 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
FD_SET(isdnfd, &set);
|
|
|
|
|
|
|
|
high_selfd = isdnfd;
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_monitor)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
if (localmonitor != -1) {
|
|
|
|
/* always watch for new connections */
|
|
|
|
FD_SET(localmonitor, &set);
|
2003-10-06 13:43:27 +04:00
|
|
|
if (localmonitor > high_selfd)
|
2001-01-06 16:00:10 +03:00
|
|
|
high_selfd = localmonitor;
|
|
|
|
}
|
|
|
|
#ifndef I4B_NOTCPIP_MONITOR
|
|
|
|
if (remotemonitor != -1) {
|
|
|
|
FD_SET(remotemonitor, &set);
|
2003-10-06 13:43:27 +04:00
|
|
|
if (remotemonitor > high_selfd)
|
2001-01-06 16:00:10 +03:00
|
|
|
high_selfd = remotemonitor;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* if there are client connections, let monitor module
|
|
|
|
* enter them into the fdset */
|
2003-10-06 13:43:27 +04:00
|
|
|
if (accepted)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
monitor_prepselect(&set, &high_selfd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
timeout.tv_sec = 1;
|
|
|
|
timeout.tv_usec = 0;
|
|
|
|
|
|
|
|
ret = select(high_selfd + 1, &set, NULL, NULL, &timeout);
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (ret > 0)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if (FD_ISSET(isdnfd, &set))
|
2001-01-06 16:00:10 +03:00
|
|
|
isdnrdhdl();
|
|
|
|
|
|
|
|
#ifdef USE_CURSES
|
2003-10-06 13:43:27 +04:00
|
|
|
if (FD_ISSET(STDIN_FILENO, &set))
|
2001-01-06 16:00:10 +03:00
|
|
|
kbdrdhdl();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef I4B_EXTERNAL_MONITOR
|
2003-10-06 13:43:27 +04:00
|
|
|
if (do_monitor)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if (localmonitor != -1 && FD_ISSET(localmonitor, &set))
|
2001-01-06 16:00:10 +03:00
|
|
|
monitor_handle_connect(localmonitor, 1);
|
|
|
|
|
|
|
|
#ifndef I4B_NOTCPIP_MONITOR
|
2003-10-06 13:43:27 +04:00
|
|
|
if (remotemonitor != -1 && FD_ISSET(remotemonitor, &set))
|
2001-01-06 16:00:10 +03:00
|
|
|
monitor_handle_connect(remotemonitor, 0);
|
|
|
|
#endif
|
2003-10-06 13:43:27 +04:00
|
|
|
if (accepted)
|
2001-01-06 16:00:10 +03:00
|
|
|
monitor_handle_input(&set);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
2003-10-06 13:43:27 +04:00
|
|
|
else if (ret == -1)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if (errno != EINTR)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "mloop: ERROR, select error on isdn device, errno = %d!", errno);
|
2001-01-06 16:00:10 +03:00
|
|
|
error_exit(1, "mloop: ERROR, select error on isdn device, errno = %d!", errno);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle timeout and recovery */
|
|
|
|
|
|
|
|
handle_recovery();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef USE_CURSES
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* data from keyboard available, read and process it
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
kbdrdhdl(void)
|
|
|
|
{
|
|
|
|
int ch = getch();
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
switch (ch)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
case 0x0c: /* control L */
|
|
|
|
wrefresh(curscr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '\n':
|
|
|
|
case '\r':
|
|
|
|
do_menu();
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* data from /dev/isdn available, read and process them
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
static void
|
|
|
|
isdnrdhdl(void)
|
|
|
|
{
|
|
|
|
static unsigned char msg_rd_buf[MSG_BUF_SIZ];
|
|
|
|
msg_hdr_t *hp = (msg_hdr_t *)&msg_rd_buf[0];
|
|
|
|
|
|
|
|
register int len;
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((len = read(isdnfd, msg_rd_buf, MSG_BUF_SIZ)) > 0)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
switch (hp->type)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
case MSG_CONNECT_IND:
|
|
|
|
msg_connect_ind((msg_connect_ind_t *)msg_rd_buf, len);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_CONNECT_ACTIVE_IND:
|
|
|
|
msg_connect_active_ind((msg_connect_active_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_DISCONNECT_IND:
|
|
|
|
msg_disconnect_ind((msg_disconnect_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_DIALOUT_IND:
|
|
|
|
msg_dialout((msg_dialout_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_ACCT_IND:
|
|
|
|
msg_accounting((msg_accounting_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_IDLE_TIMEOUT_IND:
|
|
|
|
msg_idle_timeout_ind((msg_idle_timeout_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_CHARGING_IND:
|
|
|
|
msg_charging_ind((msg_charging_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_PROCEEDING_IND:
|
|
|
|
msg_proceeding_ind((msg_proceeding_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_ALERT_IND:
|
|
|
|
msg_alert_ind((msg_alert_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_DRVRDISC_REQ:
|
|
|
|
msg_drvrdisc_req((msg_drvrdisc_req_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_L12STAT_IND:
|
|
|
|
msg_l12stat_ind((msg_l12stat_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_TEIASG_IND:
|
|
|
|
msg_teiasg_ind((msg_teiasg_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_PDEACT_IND:
|
|
|
|
msg_pdeact_ind((msg_pdeact_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_NEGCOMP_IND:
|
|
|
|
msg_negcomplete_ind((msg_negcomplete_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_IFSTATE_CHANGED_IND:
|
|
|
|
msg_ifstatechg_ind((msg_ifstatechg_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_DIALOUTNUMBER_IND:
|
|
|
|
msg_dialoutnumber((msg_dialoutnumber_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_PACKET_IND:
|
|
|
|
msg_packet_ind((msg_packet_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MSG_CONTR_EV_IND:
|
|
|
|
msg_ctrl_ev_ind((msg_ctrl_ev_ind_t *)msg_rd_buf);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
logit(LL_WRN, "ERROR, unknown message received from /dev/isdn (0x%x)", msg_rd_buf[0]);
|
|
|
|
break;
|
2001-01-06 16:00:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_WRN, "ERROR, read error on isdn device, errno = %d, length = %d", errno, len);
|
2001-01-06 16:00:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* re-read the config file on SIGHUP or menu command
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
rereadconfig(int dummy)
|
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_DMN, "re-reading configuration file");
|
2001-01-06 16:00:10 +03:00
|
|
|
|
|
|
|
close_allactive();
|
|
|
|
|
|
|
|
#if I4B_EXTERNAL_MONITOR
|
|
|
|
monitor_clear_rights();
|
|
|
|
#endif
|
|
|
|
|
2002-03-27 16:46:34 +03:00
|
|
|
remove_all_cfg_entries();
|
|
|
|
|
2001-01-06 16:00:10 +03:00
|
|
|
/* read runtime configuration file and configure ourselves */
|
|
|
|
|
|
|
|
configure(configfile, 1);
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (config_error_flag)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "rereadconfig: there were %d error(s) in the configuration file, terminating!", config_error_flag);
|
2001-01-06 16:00:10 +03:00
|
|
|
error_exit(1, "rereadconfig: there were %d error(s) in the configuration file, terminating!", config_error_flag);
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (aliasing)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
/* reread alias database */
|
|
|
|
free_aliases();
|
|
|
|
init_alias(aliasfile);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
|
|
* re-open the log/acct files on SIGUSR1
|
|
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
reopenfiles(int dummy)
|
|
|
|
{
|
2003-10-06 13:43:27 +04:00
|
|
|
if (useacctfile)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
/* close file */
|
|
|
|
|
|
|
|
fflush(acctfp);
|
|
|
|
fclose(acctfp);
|
|
|
|
|
|
|
|
/* if user specified a suffix, rename the old file */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (rotatesuffix[0] != '\0')
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
char filename[MAXPATHLEN];
|
|
|
|
|
|
|
|
snprintf(filename, sizeof(filename), "%s%s", acctfile, rotatesuffix);
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((rename(acctfile, filename)) != 0)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "reopenfiles: acct rename failed, cause = %s", strerror(errno));
|
2001-01-06 16:00:10 +03:00
|
|
|
error_exit(1, "reopenfiles: acct rename failed, cause = %s", strerror(errno));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((acctfp = fopen(acctfile, "a")) == NULL)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "ERROR, can't open acctfile %s for writing, terminating!", acctfile);
|
2001-01-06 16:00:10 +03:00
|
|
|
error_exit(1, "ERROR, can't open acctfile %s for writing, terminating!", acctfile);
|
|
|
|
}
|
|
|
|
setvbuf(acctfp, (char *)NULL, _IONBF, 0);
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (uselogfile)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
finish_log();
|
|
|
|
|
|
|
|
/* if user specified a suffix, rename the old file */
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if (rotatesuffix[0] != '\0')
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
char filename[MAXPATHLEN];
|
|
|
|
|
|
|
|
snprintf(filename, sizeof(filename), "%s%s", logfile, rotatesuffix);
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((rename(logfile, filename)) != 0)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
2002-12-06 18:00:14 +03:00
|
|
|
logit(LL_ERR, "reopenfiles: log rename failed, cause = %s", strerror(errno));
|
2001-01-06 16:00:10 +03:00
|
|
|
error_exit(1, "reopenfiles: log rename failed, cause = %s", strerror(errno));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-10-06 13:43:27 +04:00
|
|
|
if ((logfp = fopen(logfile, "a")) == NULL)
|
2001-01-06 16:00:10 +03:00
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR, cannot open logfile %s: %s\n",
|
|
|
|
logfile, strerror(errno));
|
|
|
|
error_exit(1, "reopenfiles: ERROR, cannot open logfile %s: %s\n",
|
|
|
|
logfile, strerror(errno));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set unbuffered operation */
|
|
|
|
|
|
|
|
setvbuf(logfp, (char *)NULL, _IONBF, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|