1 /* $Id: hysdn_proclog.c,v 1.9.6.3 2001/09/23 22:24:54 kai Exp $
 
   3  * Linux driver for HYSDN cards, /proc/net filesystem log functions.
 
   5  * Author    Werner Cornelius (werner@titro.de) for Hypercope GmbH
 
   6  * Copyright 1999 by Werner Cornelius (werner@titro.de)
 
   8  * This software may be used and distributed according to the terms
 
   9  * of the GNU General Public License, incorporated herein by reference.
 
  13 #include <linux/module.h>
 
  14 #include <linux/poll.h>
 
  15 #include <linux/proc_fs.h>
 
  16 #include <linux/pci.h>
 
  17 #include <linux/smp_lock.h>
 
  19 #include "hysdn_defs.h"
 
  21 /* the proc subdir for the interface is defined in the procconf module */
 
  22 extern struct proc_dir_entry *hysdn_proc_entry;
 
  24 static void put_log_buffer(hysdn_card * card, char *cp);
 
  26 /*************************************************/
 
  27 /* structure keeping ascii log for device output */
 
  28 /*************************************************/
 
  30         struct log_data *next;
 
  31         unsigned long usage_cnt;/* number of files still to work */
 
  32         void *proc_ctrl;        /* pointer to own control procdata structure */
 
  33         char log_start[2];      /* log string start (final len aligned by size) */
 
  36 /**********************************************/
 
  37 /* structure holding proc entrys for one card */
 
  38 /**********************************************/
 
  40         struct proc_dir_entry *log;     /* log entry */
 
  41         char log_name[15];      /* log filename */
 
  42         struct log_data *log_head, *log_tail;   /* head and tail for queue */
 
  43         int if_used;            /* open count for interface */
 
  44         int volatile del_lock;  /* lock for delete operations */
 
  45         unsigned char logtmp[LOG_MAX_LINELEN];
 
  46         wait_queue_head_t rd_queue;
 
  50 /**********************************************/
 
  51 /* log function for cards error log interface */
 
  52 /**********************************************/
 
  54 hysdn_card_errlog(hysdn_card * card, tErrLogEntry * logp, int maxsize)
 
  56         char buf[ERRLOG_TEXT_SIZE + 40];
 
  58         sprintf(buf, "LOG 0x%08lX 0x%08lX : %s\n", logp->ulErrType, logp->ulErrSubtype, logp->ucText);
 
  59         put_log_buffer(card, buf);      /* output the string */
 
  60 }                               /* hysdn_card_errlog */
 
  62 /***************************************************/
 
  63 /* Log function using format specifiers for output */
 
  64 /***************************************************/
 
  66 hysdn_addlog(hysdn_card * card, char *fmt,...)
 
  68         struct procdata *pd = card->proclog;
 
  73                 return;         /* log structure non existent */
 
  76         cp += sprintf(cp, "HYSDN: card %d ", card->myid);
 
  79         cp += vsprintf(cp, fmt, args);
 
  84         if (card->debug_flags & DEB_OUT_SYSLOG)
 
  85                 printk(KERN_INFO "%s", pd->logtmp);
 
  87                 put_log_buffer(card, pd->logtmp);
 
  91 /********************************************/
 
  92 /* put an log buffer into the log queue.    */
 
  93 /* This buffer will be kept until all files */
 
  94 /* opened for read got the contents.        */
 
  95 /* Flushes buffers not longer in use.       */
 
  96 /********************************************/
 
  98 put_log_buffer(hysdn_card * card, char *cp)
 
 101         struct procdata *pd = card->proclog;
 
 111         if (pd->if_used <= 0)
 
 112                 return;         /* no open file for read */
 
 114         if (!(ib = kmalloc(sizeof(struct log_data) + strlen(cp), GFP_ATOMIC)))
 
 115                  return;        /* no memory */
 
 116         strcpy(ib->log_start, cp);      /* set output string */
 
 118         ib->proc_ctrl = pd;     /* point to own control structure */
 
 119         spin_lock_irqsave(&card->hysdn_lock, flags);
 
 120         ib->usage_cnt = pd->if_used;
 
 122                 pd->log_head = ib;      /* new head */
 
 124                 pd->log_tail->next = ib;        /* follows existing messages */
 
 125         pd->log_tail = ib;      /* new tail */
 
 126         i = pd->del_lock++;     /* get lock state */
 
 127         spin_unlock_irqrestore(&card->hysdn_lock, flags);
 
 129         /* delete old entrys */
 
 131                 while (pd->log_head->next) {
 
 132                         if ((pd->log_head->usage_cnt <= 0) &&
 
 133                             (pd->log_head->next->usage_cnt <= 0)) {
 
 135                                 pd->log_head = pd->log_head->next;
 
 139                 }               /* pd->log_head->next */
 
 140         pd->del_lock--;         /* release lock level */
 
 141         wake_up_interruptible(&(pd->rd_queue));         /* announce new entry */
 
 142 }                               /* put_log_buffer */
 
 145 /******************************/
 
 146 /* file operations and tables */
 
 147 /******************************/
 
 149 /****************************************/
 
 150 /* write log file -> set log level bits */
 
 151 /****************************************/
 
 153 hysdn_log_write(struct file *file, const char __user *buf, size_t count, loff_t * off)
 
 157         unsigned char *cp, valbuf[128];
 
 159         hysdn_card *card = (hysdn_card *) file->private_data;
 
 161         if (count > (sizeof(valbuf) - 1))
 
 162                 count = sizeof(valbuf) - 1;     /* limit length */
 
 163         if (copy_from_user(valbuf, buf, count))
 
 164                 return (-EFAULT);       /* copy failed */
 
 166         valbuf[count] = 0;      /* terminating 0 */
 
 168         if ((count > 2) && (valbuf[0] == '0') && (valbuf[1] == 'x')) {
 
 169                 cp += 2;        /* pointer after hex modifier */
 
 172         /* scan the input for debug flags */
 
 174                 if ((*cp >= '0') && (*cp <= '9')) {
 
 176                         u *= base;      /* adjust to next digit */
 
 181                         break;  /* end of number */
 
 183                 if ((*cp >= 'a') && (*cp <= 'f')) {
 
 185                         u *= base;      /* adjust to next digit */
 
 186                         u += *cp++ - 'a' + 10;
 
 189                 break;          /* terminated */
 
 193                 card->debug_flags = u;  /* remember debug flags */
 
 194                 hysdn_addlog(card, "debug set to 0x%lx", card->debug_flags);
 
 197 }                               /* hysdn_log_write */
 
 203 hysdn_log_read(struct file *file, char __user *buf, size_t count, loff_t * off)
 
 205         struct log_data *inf;
 
 207         struct proc_dir_entry *pde = PDE(file->f_path.dentry->d_inode);
 
 208         struct procdata *pd = NULL;
 
 211         if (!*((struct log_data **) file->private_data)) {
 
 212                 if (file->f_flags & O_NONBLOCK)
 
 215                 /* sorry, but we need to search the card */
 
 221                         card = card->next;      /* search next entry */
 
 224                         interruptible_sleep_on(&(pd->rd_queue));
 
 229         if (!(inf = *((struct log_data **) file->private_data)))
 
 232         inf->usage_cnt--;       /* new usage count */
 
 233         file->private_data = &inf->next;        /* next structure */
 
 234         if ((len = strlen(inf->log_start)) <= count) {
 
 235                 if (copy_to_user(buf, inf->log_start, len))
 
 241 }                               /* hysdn_log_read */
 
 247 hysdn_log_open(struct inode *ino, struct file *filep)
 
 250         struct procdata *pd = NULL;
 
 257                 if (pd->log == PDE(ino))
 
 259                 card = card->next;      /* search next entry */
 
 263                 return (-ENODEV);       /* device is unknown/invalid */
 
 265         filep->private_data = card;     /* remember our own card */
 
 267         if ((filep->f_mode & (FMODE_READ | FMODE_WRITE)) == FMODE_WRITE) {
 
 268                 /* write only access -> write log level only */
 
 269         } else if ((filep->f_mode & (FMODE_READ | FMODE_WRITE)) == FMODE_READ) {
 
 271                 /* read access -> log/debug read */
 
 272                 spin_lock_irqsave(&card->hysdn_lock, flags);
 
 275                         filep->private_data = &pd->log_tail->next;
 
 277                         filep->private_data = &pd->log_head;
 
 278                 spin_unlock_irqrestore(&card->hysdn_lock, flags);
 
 279         } else {                /* simultaneous read/write access forbidden ! */
 
 281                 return (-EPERM);        /* no permission this time */
 
 284         return nonseekable_open(ino, filep);
 
 285 }                               /* hysdn_log_open */
 
 287 /*******************************************************************************/
 
 288 /* close a cardlog file. If the file has been opened for exclusive write it is */
 
 289 /* assumed as pof data input and the pof loader is noticed about.              */
 
 290 /* Otherwise file is handled as log output. In this case the interface usage   */
 
 291 /* count is decremented and all buffers are noticed of closing. If this file   */
 
 292 /* was the last one to be closed, all buffers are freed.                       */
 
 293 /*******************************************************************************/
 
 295 hysdn_log_close(struct inode *ino, struct file *filep)
 
 297         struct log_data *inf;
 
 302         spinlock_t hysdn_lock = SPIN_LOCK_UNLOCKED;
 
 305         if ((filep->f_mode & (FMODE_READ | FMODE_WRITE)) == FMODE_WRITE) {
 
 306                 /* write only access -> write debug level written */
 
 307                 retval = 0;     /* success */
 
 309                 /* read access -> log/debug read, mark one further file as closed */
 
 312                 spin_lock_irqsave(&hysdn_lock, flags);
 
 313                 inf = *((struct log_data **) filep->private_data);      /* get first log entry */
 
 315                         pd = (struct procdata *) inf->proc_ctrl;        /* still entries there */
 
 317                         /* no info available -> search card */
 
 321                                 if (pd->log == PDE(ino))
 
 323                                 card = card->next;      /* search next entry */
 
 326                                 pd = card->proclog;     /* pointer to procfs log */
 
 329                         pd->if_used--;  /* decrement interface usage count by one */
 
 332                         inf->usage_cnt--;       /* decrement usage count for buffers */
 
 335                 spin_unlock_irqrestore(&hysdn_lock, flags);
 
 338                         if (pd->if_used <= 0)   /* delete buffers if last file closed */
 
 339                                 while (pd->log_head) {
 
 341                                         pd->log_head = pd->log_head->next;
 
 348 }                               /* hysdn_log_close */
 
 350 /*************************************************/
 
 351 /* select/poll routine to be able using select() */
 
 352 /*************************************************/
 
 354 hysdn_log_poll(struct file *file, poll_table * wait)
 
 356         unsigned int mask = 0;
 
 357         struct proc_dir_entry *pde = PDE(file->f_path.dentry->d_inode);
 
 359         struct procdata *pd = NULL;
 
 361         if ((file->f_mode & (FMODE_READ | FMODE_WRITE)) == FMODE_WRITE)
 
 362                 return (mask);  /* no polling for write supported */
 
 364         /* we need to search the card */
 
 370                 card = card->next;      /* search next entry */
 
 373                 return (mask);  /* card not found */
 
 375         poll_wait(file, &(pd->rd_queue), wait);
 
 377         if (*((struct log_data **) file->private_data))
 
 378                 mask |= POLLIN | POLLRDNORM;
 
 381 }                               /* hysdn_log_poll */
 
 383 /**************************************************/
 
 384 /* table for log filesystem functions defined above. */
 
 385 /**************************************************/
 
 386 static const struct file_operations log_fops =
 
 389         .read           = hysdn_log_read,
 
 390         .write          = hysdn_log_write,
 
 391         .poll           = hysdn_log_poll,
 
 392         .open           = hysdn_log_open,
 
 393         .release        = hysdn_log_close,                                        
 
 397 /***********************************************************************************/
 
 398 /* hysdn_proclog_init is called when the module is loaded after creating the cards */
 
 400 /***********************************************************************************/
 
 402 hysdn_proclog_init(hysdn_card * card)
 
 406         /* create a cardlog proc entry */
 
 408         if ((pd = kzalloc(sizeof(struct procdata), GFP_KERNEL)) != NULL) {
 
 409                 sprintf(pd->log_name, "%s%d", PROC_LOG_BASENAME, card->myid);
 
 410                 if ((pd->log = create_proc_entry(pd->log_name, S_IFREG | S_IRUGO | S_IWUSR, hysdn_proc_entry)) != NULL) {
 
 411                         pd->log->proc_fops = &log_fops; 
 
 412                         pd->log->owner = THIS_MODULE;
 
 415                 init_waitqueue_head(&(pd->rd_queue));
 
 417                 card->proclog = (void *) pd;    /* remember procfs structure */
 
 420 }                               /* hysdn_proclog_init */
 
 422 /************************************************************************************/
 
 423 /* hysdn_proclog_release is called when the module is unloaded and before the cards */
 
 424 /* conf file is released                                                            */
 
 425 /* The module counter is assumed to be 0 !                                          */
 
 426 /************************************************************************************/
 
 428 hysdn_proclog_release(hysdn_card * card)
 
 432         if ((pd = (struct procdata *) card->proclog) != NULL) {
 
 434                         remove_proc_entry(pd->log_name, hysdn_proc_entry);
 
 435                 kfree(pd);      /* release memory */
 
 436                 card->proclog = NULL;
 
 438 }                               /* hysdn_proclog_release */