diff options
| -rw-r--r-- | plugin_isdn.c | 248 | 
1 files changed, 248 insertions, 0 deletions
| diff --git a/plugin_isdn.c b/plugin_isdn.c new file mode 100644 index 0000000..d616efc --- /dev/null +++ b/plugin_isdn.c @@ -0,0 +1,248 @@ +/* $Id: plugin_isdn.c,v 1.1 2004/05/19 05:23:25 reinelt Exp $ + * + * plugin for ISDN subsystem + * + * Copyright 2003 Michael Reinelt <reinelt@eunet.at> + * Copyright 2004 The LCD4Linux Team <lcd4linux-devel@users.sourceforge.net> + * + * Based on the old isdn client (isdn.c) which is + * Copyright 1999, 2000 Michael Reinelt <reinelt@eunet.at> + * + * This file is part of LCD4Linux. + * + * LCD4Linux is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2, or (at your option) + * any later version. + * + * LCD4Linux is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + * + * + * $Log: plugin_isdn.c,v $ + * Revision 1.1  2004/05/19 05:23:25  reinelt + * + * plugin_isdn.c added (sorry, I forgot...) + * + */ + +/*  + * exported functions: + * + * int plugin_init_isdn (void) + *  adds functions to access ISDN information + * + */ + + +#include "config.h" + +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> +#include <errno.h> +#include <fcntl.h> +#include <sys/ioctl.h> +#include <sys/types.h> + +#ifdef HAVE_LINUX_ISDN_H +#include <linux/isdn.h> +#else +#warning isdn.h not found. CPS support deactivated. +#endif + +#include "debug.h" +#include "plugin.h" +#include "qprintf.h" +#include "hash.h" + + +typedef struct { +  unsigned long in; +  unsigned long out; +} CPS; + + +static HASH ISDN_INFO = { 0, }; +static HASH ISDN_CPS  = { 0, }; + + +static void hash_set_info (char *name, int channel, char *val) +{ +  char key[16]; + +  qprintf (key, sizeof(key), "%s[%d]", name, channel); +  hash_set (&ISDN_INFO, key, val); +} + +static int parse_isdninfo (void) +{ +  int age; +  FILE *stream; +  long flags; +   +  // reread every 10 msec only +  age = hash_age(&ISDN_INFO, NULL, NULL); +  if (age > 0 && age <= 10) return 0; + +  // open file +  stream = fopen ("/dev/isdninfo", "r"); +  if (stream == NULL) { +    error ("open(/dev/isdninfo) failed: %s", strerror(errno)); +    return -1; +  } +   +  // get flags +  flags = fcntl(fileno(stream), F_GETFL); +  if (flags < 0) { +    error ("fcntl(/dev/isdninfo, F_GETFL) failed: %s", strerror(errno)); +    return -1; +  } +   +  // set O_NONBLOCK +  if (fcntl (fileno(stream), F_SETFL, flags | O_NONBLOCK) < 0) { +    error ("fcntl(/dev/isdninfo, F_SETFL, O_NONBLOCK) failed: %s", strerror(errno)); +    return -1; +  } + +  while (!feof(stream)) { +    char buffer[4096]; +    char *beg, *end; +    if (fgets (buffer, sizeof(buffer), stream) == NULL) break; +    beg = strchr(buffer, ':'); +    if (beg!=NULL) { +      char delim[] = " \t\n"; +      int i = 0; +      *beg++ = '\0'; +      while (*beg && strchr(delim, *beg)) beg++;  +      while (beg && *beg) { +	if ((end = strpbrk(beg, delim))) *end = '\0';  +	hash_set_info(buffer, i, beg); +	beg = end ? end+1 : NULL; +	while (*beg && strchr(delim, *beg)) beg++;  +	i++; +      } +    } else { +      error ("Huh? no colon found in <%s>", buffer); +    } +  } + +  fclose (stream); + +  return 0; +}   + + +static void my_isdn_info (RESULT *result, RESULT *arg1, RESULT *arg2) +{ +  char key[16], *val; + +  if (parse_isdninfo()<0) { +    SetResult(&result, R_STRING, "");  +    return; +  } +   +  qprintf(key, sizeof(key), "%s[%d]", R2S(arg1), (int)R2N(arg2)); +  val = hash_get(&ISDN_INFO, key); +  if (val == NULL) val = ""; +  SetResult(&result, R_STRING, val);  +} + + +#ifdef HAVE_LINUX_ISDN_H + +static void hash_set_cps (int channel, CPS *cps) +{ +  char key[16], val[16]; + +  qprintf (key, sizeof(key), channel < 0 ? "i" : "i%d", channel); +  qprintf (val, sizeof(val), "%u", cps->in); +  hash_set_delta (&ISDN_CPS, key, val); + +  qprintf (key, sizeof(key), channel < 0 ? "o" : "o%d", channel); +  qprintf (val, sizeof(val), "%u", cps->out); +  hash_set_delta (&ISDN_CPS, key, val); +} + + +static int get_cps(void) +{ +  int age, i; +  static int fd = -2; +  CPS cps[ISDN_MAX_CHANNELS]; +  CPS sum; + +  // reread every 10 msec only +  age = hash_age(&ISDN_CPS, NULL, NULL); +  if (age > 0 && age <= 10) return 0; +   +  if (fd == -1) return -1; +   +  if (fd == -2) { +    fd = open("/dev/isdninfo", O_RDONLY | O_NDELAY); +    if (fd == -1) { +      error ("open(/dev/isdninfo) failed: %s", strerror(errno)); +      return -1; +    } +  } +   +  if (ioctl(fd, IIOCGETCPS, &cps)) { +    error("ioctl(IIOCGETCPS) failed: %s", strerror(errno)); +    fd = -1; +    return -1; +  } + +  sum.in  = 0; +  sum.out = 0; +  for (i = 0; i < ISDN_MAX_CHANNELS; i++) { +    sum.in  += cps[i].in; +    sum.out += cps[i].out; +    hash_set_cps (i, &cps[i]); +  } +  hash_set_cps (-1, &sum); + +  return 0; +} + + +static void my_isdn_cps (RESULT *result, RESULT *arg1, RESULT *arg2) +{ +  double value; +   +  if (get_cps()<0) { +    SetResult(&result, R_STRING, "");  +    return; +  } +   +  value = hash_get_delta(&ISDN_CPS, R2S(arg1), R2N(arg2)); +  SetResult(&result, R_NUMBER, &value);  + +} + +#endif + + +int plugin_init_isdn (void) +{ +  AddFunction ("isdn::info", 2, my_isdn_info); + +#ifdef HAVE_LINUX_ISDN_H +  AddFunction ("isdn::cps",  2, my_isdn_cps); +#endif + +  return 0; +} + + +void plugin_exit_isdn(void)  +{ +  hash_destroy(&ISDN_INFO); +  hash_destroy(&ISDN_CPS); +} | 
