/* $Id: plugin_isdn.c,v 1.7 2005/05/08 04:32:44 reinelt Exp $ * * plugin for ISDN subsystem * * Copyright (C) 2003 Michael Reinelt * Copyright (C) 2004 The LCD4Linux Team * * Based on the old isdn client (isdn.c) which is * Copyright (C) 1999, 2000 Michael Reinelt * * 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.7 2005/05/08 04:32:44 reinelt * CodingStyle added and applied * * Revision 1.6 2005/01/18 06:30:23 reinelt * added (C) to all copyright statements * * Revision 1.5 2004/06/26 12:05:00 reinelt * * uh-oh... the last CVS log message messed up things a lot... * * Revision 1.4 2004/06/26 09:27:21 reinelt * * added '-W' to CFLAGS * changed all C++ comments to C ones * cleaned up a lot of signed/unsigned mistakes * * Revision 1.3 2004/06/20 10:09:56 reinelt * * 'const'ified the whole source * * Revision 1.2 2004/06/17 06:23:43 reinelt * * hash handling rewritten to solve performance issues * * 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 #include #include #include #include #include #include #include #ifdef HAVE_LINUX_ISDN_H #include #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; static HASH ISDN_CPS; static void hash_put_info(const char *name, const int channel, const char *val) { char key[16]; qprintf(key, sizeof(key), "%s[%d]", name, channel); hash_put(&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); 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_put_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, NULL); if (val == NULL) val = ""; SetResult(&result, R_STRING, val); } #ifdef HAVE_LINUX_ISDN_H static void hash_put_cps(const int channel, const 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_put_delta(&ISDN_CPS, key, val); qprintf(key, sizeof(key), channel < 0 ? "o" : "o%d", channel); qprintf(val, sizeof(val), "%u", cps->out); hash_put_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); 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_put_cps(i, &cps[i]); } hash_put_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), NULL, R2N(arg2)); SetResult(&result, R_NUMBER, &value); } #endif int plugin_init_isdn(void) { hash_create(&ISDN_INFO); hash_create(&ISDN_CPS); 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); } r: #888888; font-weight: bold } /* Keyword.Type */ .highlight .m { color: #0000DD; font-weight: bold } /* Literal.Number */ .highlight .s { color: #dd2200; background-color: #fff0f0 } /* Literal.String */ .highlight .na { color: #336699 } /* Name.Attribute */ .highlight .nb { color: #003388 } /* Name.Builtin */ .highlight .nc { color: #bb0066; font-weight: bold } /* Name.Class */ .highlight .no { color: #003366; font-weight: bold } /* Name.Constant */ .highlight .nd { color: #555555 } /* Name.Decorator */ .highlight .ne { color: #bb0066; font-weight: bold } /* Name.Exception */ .highlight .nf { color: #0066bb; font-weight: bold } /* Name.Function */ .highlight .nl { color: #336699; font-style: italic } /* Name.Label */ .highlight .nn { color: #bb0066; font-weight: bold } /* Name.Namespace */ .highlight .py { color: #336699; font-weight: bold } /* Name.Property */ .highlight .nt { color: #bb0066; font-weight: bold } /* Name.Tag */ .highlight .nv { color: #336699 } /* Name.Variable */ .highlight .ow { color: #008800 } /* Operator.Word */ .highlight .w { color: #bbbbbb } /* Text.Whitespace */ .highlight .mb { color: #0000DD; font-weight: bold } /* Literal.Number.Bin */ .highlight .mf { color: #0000DD; font-weight: bold } /* Literal.Number.Float */ .highlight .mh { color: #0000DD; font-weight: bold } /* Literal.Number.Hex */ .highlight .mi { color: #0000DD; font-weight: bold } /* Literal.Number.Integer */ .highlight .mo { color: #0000DD; font-weight: bold } /* Literal.Number.Oct */ .highlight .sa { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Affix */ .highlight .sb { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Backtick */ .highlight .sc { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Char */ .highlight .dl { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Delimiter */ .highlight .sd { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Doc */ .highlight .s2 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Double */ .highlight .se { color: #0044dd; background-color: #fff0f0 } /* Literal.String.Escape */ .highlight .sh { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Heredoc */ .highlight .si { color: #3333bb; background-color: #fff0f0 } /* Literal.String.Interpol */ .highlight .sx { color: #22bb22; background-color: #f0fff0 } /* Literal.String.Other */ .highlight .sr { color: #008800; background-color: #fff0ff } /* Literal.String.Regex */ .highlight .s1 { color: #dd2200; background-color: #fff0f0 } /* Literal.String.Single */ .highlight .ss { color: #aa6600; background-color: #fff0f0 } /* Literal.String.Symbol */ .highlight .bp { color: #003388 } /* Name.Builtin.Pseudo */ .highlight .fm { color: #0066bb; font-weight: bold } /* Name.Function.Magic */ .highlight .vc { color: #336699 } /* Name.Variable.Class */ .highlight .vg { color: #dd7700 } /* Name.Variable.Global */ .highlight .vi { color: #3333bb } /* Name.Variable.Instance */ .highlight .vm { color: #336699 } /* Name.Variable.Magic */ .highlight .il { color: #0000DD; font-weight: bold } /* Literal.Number.Integer.Long */
   ---   LCD4Linux News   ---



lcd4linux-0.98

- improved signal-handling: lcd4linux can be restarted by sending a SIGHUP

- lcd4linux uses syslog for messages/warnings etc.

- support for GPO's (general purpose outputs, e.g. to connect a LED)
  The MatroxOrbital Displays have one GPO, and you can connect up to eight
  LED's to the parallel port using two IC's (see README.HD44780.GPO)

- new delay loop for parallel port displays: uses either gettimeofday() or
  rdtsc (time stamp counter on newer processors). This delay needs no longer
  a calibration! The 'Delay'-entry in lcd4linux.conf has gone.

- new raster driver for PNG creation

- new curses-based text driver

- security: lcd4linux.conf must have the same user/group than the executable, 
  and must not me accesible by group and other (i.e. mode 0600). This is 
  necessary because lcd4linux.conf can contain usernames and passwords for 
  POP and IMAP queries.

- support for plugins: new token 'x1'..'x9' for externel programs

- new token 'ic' (ISDN connected/offline, mainly for a LED connected to a GPO)

- APM client: new tokens 'bp', 'bs', 'bd' (battery percentage, status and 
  duration)

- mail client: new tokens 'e1'..'e9' for counting emails in local mailboxes

- new configure-options '--with-drivers=<list>', so you can compile a minimal
  executable which contains only the drivers you need.

- seti@home client: new tokens 'hc' (% completed) and 'ht' (cpu time needed)

- simple web-server driver (see README.Webinterface)

- uses new (kernel 2.4) ppdev system to access parallel port via ioctl()

- mail suport for pop3 and imap4, token 'u1' ..'u9' show unseen mail
  (which is propably incorrect for normal mbox)

- added another verbose level (-vvv) for debugging socket traffic

- added nntp support for count of news/unread messages in subscribed NGs

- implemented time series bar '$t'