1 /* net/atm/proc.c - ATM /proc interface
 
   3  * Written 1995-2000 by Werner Almesberger, EPFL LRC/ICA
 
   5  * seq_file api usage by romieu@fr.zoreil.com
 
   7  * Evaluating the efficiency of the whole thing if left as an exercise to
 
  11 #include <linux/module.h> /* for EXPORT_SYMBOL */
 
  12 #include <linux/string.h>
 
  13 #include <linux/types.h>
 
  16 #include <linux/stat.h>
 
  17 #include <linux/proc_fs.h>
 
  18 #include <linux/seq_file.h>
 
  19 #include <linux/errno.h>
 
  20 #include <linux/atm.h>
 
  21 #include <linux/atmdev.h>
 
  22 #include <linux/netdevice.h>
 
  23 #include <linux/atmclip.h>
 
  24 #include <linux/init.h> /* for __init */
 
  25 #include <net/atmclip.h>
 
  26 #include <asm/uaccess.h>
 
  27 #include <asm/atomic.h>
 
  28 #include <asm/param.h> /* for HZ */
 
  29 #include "resources.h"
 
  30 #include "common.h" /* atm_proc_init prototype */
 
  31 #include "signaling.h" /* to get sigd - ugly too */
 
  33 static ssize_t proc_dev_atm_read(struct file *file,char __user *buf,size_t count,
 
  36 static struct file_operations proc_atm_dev_ops = {
 
  38         .read =         proc_dev_atm_read,
 
  41 static void add_stats(struct seq_file *seq, const char *aal,
 
  42   const struct k_atm_aal_stats *stats)
 
  44         seq_printf(seq, "%s ( %d %d %d %d %d )", aal,
 
  45             atomic_read(&stats->tx),atomic_read(&stats->tx_err),
 
  46             atomic_read(&stats->rx),atomic_read(&stats->rx_err),
 
  47             atomic_read(&stats->rx_drop));
 
  50 static void atm_dev_info(struct seq_file *seq, const struct atm_dev *dev)
 
  54         seq_printf(seq, "%3d %-8s", dev->number, dev->type);
 
  55         for (i = 0; i < ESI_LEN; i++)
 
  56                 seq_printf(seq, "%02x", dev->esi[i]);
 
  58         add_stats(seq, "0", &dev->stats.aal0);
 
  60         add_stats(seq, "5", &dev->stats.aal5);
 
  61         seq_printf(seq, "\t[%d]", atomic_read(&dev->refcnt));
 
  71 static inline int compare_family(struct sock *sk, int family)
 
  73         return !family || (sk->sk_family == family);
 
  76 static int __vcc_walk(struct sock **sock, int family, int *bucket, loff_t l)
 
  78         struct sock *sk = *sock;
 
  80         if (sk == (void *)1) {
 
  81                 for (*bucket = 0; *bucket < VCC_HTABLE_SIZE; ++*bucket) {
 
  82                         struct hlist_head *head = &vcc_hash[*bucket];
 
  84                         sk = hlist_empty(head) ? NULL : __sk_head(head);
 
  91         for (; sk; sk = sk_next(sk)) {
 
  92                 l -= compare_family(sk, family);
 
  96         if (!sk && ++*bucket < VCC_HTABLE_SIZE) {
 
  97                 sk = sk_head(&vcc_hash[*bucket]);
 
 106 static inline void *vcc_walk(struct vcc_state *state, loff_t l)
 
 108         return __vcc_walk(&state->sk, state->family, &state->bucket, l) ?
 
 112 static int __vcc_seq_open(struct inode *inode, struct file *file,
 
 113         int family, struct seq_operations *ops)
 
 115         struct vcc_state *state;
 
 116         struct seq_file *seq;
 
 119         state = kmalloc(sizeof(*state), GFP_KERNEL);
 
 123         rc = seq_open(file, ops);
 
 127         state->family = family;
 
 129         seq = file->private_data;
 
 130         seq->private = state;
 
 138 static int vcc_seq_release(struct inode *inode, struct file *file)
 
 140         return seq_release_private(inode, file);
 
 143 static void *vcc_seq_start(struct seq_file *seq, loff_t *pos)
 
 145         struct vcc_state *state = seq->private;
 
 148         read_lock(&vcc_sklist_lock);
 
 149         state->sk = (void *)1;
 
 150         return left ? vcc_walk(state, left) : (void *)1;
 
 153 static void vcc_seq_stop(struct seq_file *seq, void *v)
 
 155         read_unlock(&vcc_sklist_lock);
 
 158 static void *vcc_seq_next(struct seq_file *seq, void *v, loff_t *pos)
 
 160         struct vcc_state *state = seq->private;
 
 162         v = vcc_walk(state, 1);
 
 163         *pos += !!PTR_ERR(v);
 
 167 static void pvc_info(struct seq_file *seq, struct atm_vcc *vcc)
 
 169         static const char *class_name[] = { "off","UBR","CBR","VBR","ABR" };
 
 170         static const char *aal_name[] = {
 
 171                 "---",  "1",    "2",    "3/4",  /*  0- 3 */
 
 172                 "???",  "5",    "???",  "???",  /*  4- 7 */
 
 173                 "???",  "???",  "???",  "???",  /*  8-11 */
 
 174                 "???",  "0",    "???",  "???"}; /* 12-15 */
 
 176         seq_printf(seq, "%3d %3d %5d %-3s %7d %-5s %7d %-6s",
 
 177             vcc->dev->number,vcc->vpi,vcc->vci,
 
 178             vcc->qos.aal >= sizeof(aal_name)/sizeof(aal_name[0]) ? "err" :
 
 179             aal_name[vcc->qos.aal],vcc->qos.rxtp.min_pcr,
 
 180             class_name[vcc->qos.rxtp.traffic_class],vcc->qos.txtp.min_pcr,
 
 181             class_name[vcc->qos.txtp.traffic_class]);
 
 182         if (test_bit(ATM_VF_IS_CLIP, &vcc->flags)) {
 
 183                 struct clip_vcc *clip_vcc = CLIP_VCC(vcc);
 
 184                 struct net_device *dev;
 
 186                 dev = clip_vcc->entry ? clip_vcc->entry->neigh->dev : NULL;
 
 187                 seq_printf(seq, "CLIP, Itf:%s, Encap:",
 
 188                     dev ? dev->name : "none?");
 
 189                 seq_printf(seq, "%s", clip_vcc->encap ? "LLC/SNAP" : "None");
 
 194 static const char *vcc_state(struct atm_vcc *vcc)
 
 196         static const char *map[] = { ATM_VS2TXT_MAP };
 
 198         return map[ATM_VF2VS(vcc->flags)];
 
 201 static void vcc_info(struct seq_file *seq, struct atm_vcc *vcc)
 
 203         struct sock *sk = sk_atm(vcc);
 
 205         seq_printf(seq, "%p ", vcc);
 
 207                 seq_printf(seq, "Unassigned    ");
 
 209                 seq_printf(seq, "%3d %3d %5d ", vcc->dev->number, vcc->vpi,
 
 211         switch (sk->sk_family) {
 
 213                         seq_printf(seq, "PVC");
 
 216                         seq_printf(seq, "SVC");
 
 219                         seq_printf(seq, "%3d", sk->sk_family);
 
 221         seq_printf(seq, " %04lx  %5d %7d/%7d %7d/%7d [%d]\n", vcc->flags, sk->sk_err,
 
 222                   atomic_read(&sk->sk_wmem_alloc), sk->sk_sndbuf,
 
 223                   atomic_read(&sk->sk_rmem_alloc), sk->sk_rcvbuf,
 
 224                   atomic_read(&sk->sk_refcnt));
 
 227 static void svc_info(struct seq_file *seq, struct atm_vcc *vcc)
 
 230                 seq_printf(seq, sizeof(void *) == 4 ?
 
 231                            "N/A@%p%10s" : "N/A@%p%2s", vcc, "");
 
 233                 seq_printf(seq, "%3d %3d %5d         ",
 
 234                            vcc->dev->number, vcc->vpi, vcc->vci);
 
 235         seq_printf(seq, "%-10s ", vcc_state(vcc));
 
 236         seq_printf(seq, "%s%s", vcc->remote.sas_addr.pub,
 
 237             *vcc->remote.sas_addr.pub && *vcc->remote.sas_addr.prv ? "+" : "");
 
 238         if (*vcc->remote.sas_addr.prv) {
 
 241                 for (i = 0; i < ATM_ESA_LEN; i++)
 
 242                         seq_printf(seq, "%02x", vcc->remote.sas_addr.prv[i]);
 
 247 static int atm_dev_seq_show(struct seq_file *seq, void *v)
 
 249         static char atm_dev_banner[] =
 
 250                 "Itf Type    ESI/\"MAC\"addr "
 
 251                 "AAL(TX,err,RX,err,drop) ...               [refcnt]\n";
 
 254                 seq_puts(seq, atm_dev_banner);
 
 256                 struct atm_dev *dev = list_entry(v, struct atm_dev, dev_list);
 
 258                 atm_dev_info(seq, dev);
 
 263 static struct seq_operations atm_dev_seq_ops = {
 
 264         .start  = atm_dev_seq_start,
 
 265         .next   = atm_dev_seq_next,
 
 266         .stop   = atm_dev_seq_stop,
 
 267         .show   = atm_dev_seq_show,
 
 270 static int atm_dev_seq_open(struct inode *inode, struct file *file)
 
 272         return seq_open(file, &atm_dev_seq_ops);
 
 275 static struct file_operations devices_seq_fops = {
 
 276         .open           = atm_dev_seq_open,
 
 279         .release        = seq_release,
 
 282 static int pvc_seq_show(struct seq_file *seq, void *v)
 
 284         static char atm_pvc_banner[] = 
 
 285                 "Itf VPI VCI   AAL RX(PCR,Class) TX(PCR,Class)\n";
 
 288                 seq_puts(seq, atm_pvc_banner);
 
 290                 struct vcc_state *state = seq->private;
 
 291                 struct atm_vcc *vcc = atm_sk(state->sk);
 
 298 static struct seq_operations pvc_seq_ops = {
 
 299         .start  = vcc_seq_start,
 
 300         .next   = vcc_seq_next,
 
 301         .stop   = vcc_seq_stop,
 
 302         .show   = pvc_seq_show,
 
 305 static int pvc_seq_open(struct inode *inode, struct file *file)
 
 307         return __vcc_seq_open(inode, file, PF_ATMPVC, &pvc_seq_ops);
 
 310 static struct file_operations pvc_seq_fops = {
 
 311         .open           = pvc_seq_open,
 
 314         .release        = vcc_seq_release,
 
 317 static int vcc_seq_show(struct seq_file *seq, void *v)
 
 319         if (v == (void *)1) {
 
 320                 seq_printf(seq, sizeof(void *) == 4 ? "%-8s%s" : "%-16s%s",
 
 321                         "Address ", "Itf VPI VCI   Fam Flags Reply "
 
 322                         "Send buffer     Recv buffer      [refcnt]\n");
 
 324                 struct vcc_state *state = seq->private;
 
 325                 struct atm_vcc *vcc = atm_sk(state->sk);
 
 332 static struct seq_operations vcc_seq_ops = {
 
 333         .start  = vcc_seq_start,
 
 334         .next   = vcc_seq_next,
 
 335         .stop   = vcc_seq_stop,
 
 336         .show   = vcc_seq_show,
 
 339 static int vcc_seq_open(struct inode *inode, struct file *file)
 
 341         return __vcc_seq_open(inode, file, 0, &vcc_seq_ops);
 
 344 static struct file_operations vcc_seq_fops = {
 
 345         .open           = vcc_seq_open,
 
 348         .release        = vcc_seq_release,
 
 351 static int svc_seq_show(struct seq_file *seq, void *v)
 
 353         static char atm_svc_banner[] = 
 
 354                 "Itf VPI VCI           State      Remote\n";
 
 357                 seq_puts(seq, atm_svc_banner);
 
 359                 struct vcc_state *state = seq->private;
 
 360                 struct atm_vcc *vcc = atm_sk(state->sk);
 
 367 static struct seq_operations svc_seq_ops = {
 
 368         .start  = vcc_seq_start,
 
 369         .next   = vcc_seq_next,
 
 370         .stop   = vcc_seq_stop,
 
 371         .show   = svc_seq_show,
 
 374 static int svc_seq_open(struct inode *inode, struct file *file)
 
 376         return __vcc_seq_open(inode, file, PF_ATMSVC, &svc_seq_ops);
 
 379 static struct file_operations svc_seq_fops = {
 
 380         .open           = svc_seq_open,
 
 383         .release        = vcc_seq_release,
 
 386 static ssize_t proc_dev_atm_read(struct file *file, char __user *buf,
 
 387                                  size_t count, loff_t *pos)
 
 393         if (count == 0) return 0;
 
 394         page = get_zeroed_page(GFP_KERNEL);
 
 395         if (!page) return -ENOMEM;
 
 396         dev = PDE(file->f_dentry->d_inode)->data;
 
 397         if (!dev->ops->proc_read)
 
 400                 length = dev->ops->proc_read(dev,pos,(char *) page);
 
 401                 if (length > count) length = -EINVAL;
 
 404                 if (copy_to_user(buf,(char *) page,length)) length = -EFAULT;
 
 412 struct proc_dir_entry *atm_proc_root;
 
 413 EXPORT_SYMBOL(atm_proc_root);
 
 416 int atm_proc_dev_register(struct atm_dev *dev)
 
 422         if (!dev->ops->proc_read)
 
 427         for (num = dev->number; num; num /= 10) digits++;
 
 428         if (!digits) digits++;
 
 430         dev->proc_name = kmalloc(strlen(dev->type) + digits + 2, GFP_KERNEL);
 
 433         sprintf(dev->proc_name,"%s:%d",dev->type, dev->number);
 
 435         dev->proc_entry = create_proc_entry(dev->proc_name, 0, atm_proc_root);
 
 436         if (!dev->proc_entry)
 
 438         dev->proc_entry->data = dev;
 
 439         dev->proc_entry->proc_fops = &proc_atm_dev_ops;
 
 440         dev->proc_entry->owner = THIS_MODULE;
 
 443         kfree(dev->proc_name);
 
 449 void atm_proc_dev_deregister(struct atm_dev *dev)
 
 451         if (!dev->ops->proc_read)
 
 454         remove_proc_entry(dev->proc_name, atm_proc_root);
 
 455         kfree(dev->proc_name);
 
 458 static struct atm_proc_entry {
 
 460         struct file_operations *proc_fops;
 
 461         struct proc_dir_entry *dirent;
 
 462 } atm_proc_ents[] = {
 
 463         { .name = "devices",    .proc_fops = &devices_seq_fops },
 
 464         { .name = "pvc",        .proc_fops = &pvc_seq_fops },
 
 465         { .name = "svc",        .proc_fops = &svc_seq_fops },
 
 466         { .name = "vc",         .proc_fops = &vcc_seq_fops },
 
 467         { .name = NULL,         .proc_fops = NULL }
 
 470 static void atm_proc_dirs_remove(void)
 
 472         static struct atm_proc_entry *e;
 
 474         for (e = atm_proc_ents; e->name; e++) {
 
 476                         remove_proc_entry(e->name, atm_proc_root);
 
 478         remove_proc_entry("net/atm", NULL);
 
 481 int __init atm_proc_init(void)
 
 483         static struct atm_proc_entry *e;
 
 486         atm_proc_root = proc_mkdir("net/atm",NULL);
 
 489         for (e = atm_proc_ents; e->name; e++) {
 
 490                 struct proc_dir_entry *dirent;
 
 492                 dirent = create_proc_entry(e->name, S_IRUGO, atm_proc_root);
 
 495                 dirent->proc_fops = e->proc_fops;
 
 496                 dirent->owner = THIS_MODULE;
 
 504         atm_proc_dirs_remove();
 
 510 void atm_proc_exit(void)
 
 512         atm_proc_dirs_remove();