J. Bruce Fields wrote:
> On Wed, Sep 19, 2007 at 03:35:27PM +0400, Pavel Emelyanov wrote:
>> Currently /proc/locks is shown with a proc_read function, but
>> its behavior is rather complex as it has to manually handle
>> current offset and buffer length. On the other hand, files 
>> that show objects from lists can be easily reimplemented using
>> the sequential files and the seq_list_XXX() helpers.
>>
>> This saves (as usually) 16 lines of code and more than 200 from
>> the .text section.
>>
>> This patch looks rather ugly, as diff often uses curly braces
>> as not-changed lines, but I haven't managed to organize the 
>> code to make diff look better. Except for move the whole proc
>> related stuff upper/lower in the locks.c file...
> 
> Fine by me.  I've lost track--are you assuming some earlier patches are
> applied?  It doesn't seem to apply to any copy of locks.c I have.

Yes, this applies the the set of patches that cleanup the 
MANDATORY_LOCK macro (5 patches just accepted by Andrew).

> --b.
> 
>> Signed-off-by: Pavel Emelyanov <[EMAIL PROTECTED]>
>>
>> ---
>>
>>  fs/locks.c          |  124 
>> ++++++++++++++++++++++------------------------------
>>  fs/proc/proc_misc.c |   20 ++++----
>>  2 files changed, 64 insertions(+), 80 deletions(-)
>>
>> diff --git a/fs/locks.c b/fs/locks.c
>> index 8e849ed..746dc70 100644
>> --- a/fs/locks.c
>> +++ b/fs/locks.c
>> @@ -2032,134 +2032,116 @@ int vfs_cancel_lock(struct file *filp, s
>>  
>>  EXPORT_SYMBOL_GPL(vfs_cancel_lock);
>>  
>> -static void lock_get_status(char* out, struct file_lock *fl, int id, char 
>> *pfx)
>> +#ifdef CONFIG_PROC_FS
>> +#include <linux/seq_file.h>
>> +
>> +static void lock_get_status(struct seq_file *f, struct file_lock *fl,
>> +                                                    int id, char *pfx)
>>  {
>>      struct inode *inode = NULL;
>>  
>>      if (fl->fl_file != NULL)
>>              inode = fl->fl_file->f_path.dentry->d_inode;
>>  
>> -    out += sprintf(out, "%d:%s ", id, pfx);
>> +    seq_printf(f, "%d:%s ", id, pfx);
>>      if (IS_POSIX(fl)) {
>> -            out += sprintf(out, "%6s %s ",
>> +            seq_printf(f, "%6s %s ",
>>                           (fl->fl_flags & FL_ACCESS) ? "ACCESS" : "POSIX ",
>>                           (inode == NULL) ? "*NOINODE*" :
>>                           mandatory_lock(inode) ? "MANDATORY" : "ADVISORY ");
>>      } else if (IS_FLOCK(fl)) {
>>              if (fl->fl_type & LOCK_MAND) {
>> -                    out += sprintf(out, "FLOCK  MSNFS     ");
>> +                    seq_printf(f, "FLOCK  MSNFS     ");
>>              } else {
>> -                    out += sprintf(out, "FLOCK  ADVISORY  ");
>> +                    seq_printf(f, "FLOCK  ADVISORY  ");
>>              }
>>      } else if (IS_LEASE(fl)) {
>> -            out += sprintf(out, "LEASE  ");
>> +            seq_printf(f, "LEASE  ");
>>              if (fl->fl_type & F_INPROGRESS)
>> -                    out += sprintf(out, "BREAKING  ");
>> +                    seq_printf(f, "BREAKING  ");
>>              else if (fl->fl_file)
>> -                    out += sprintf(out, "ACTIVE    ");
>> +                    seq_printf(f, "ACTIVE    ");
>>              else
>> -                    out += sprintf(out, "BREAKER   ");
>> +                    seq_printf(f, "BREAKER   ");
>>      } else {
>> -            out += sprintf(out, "UNKNOWN UNKNOWN  ");
>> +            seq_printf(f, "UNKNOWN UNKNOWN  ");
>>      }
>>      if (fl->fl_type & LOCK_MAND) {
>> -            out += sprintf(out, "%s ",
>> +            seq_printf(f, "%s ",
>>                             (fl->fl_type & LOCK_READ)
>>                             ? (fl->fl_type & LOCK_WRITE) ? "RW   " : "READ "
>>                             : (fl->fl_type & LOCK_WRITE) ? "WRITE" : "NONE 
>> ");
>>      } else {
>> -            out += sprintf(out, "%s ",
>> +            seq_printf(f, "%s ",
>>                             (fl->fl_type & F_INPROGRESS)
>>                             ? (fl->fl_type & F_UNLCK) ? "UNLCK" : "READ "
>>                             : (fl->fl_type & F_WRLCK) ? "WRITE" : "READ ");
>>      }
>>      if (inode) {
>>  #ifdef WE_CAN_BREAK_LSLK_NOW
>> -            out += sprintf(out, "%d %s:%ld ", fl->fl_pid,
>> +            seq_printf(f, "%d %s:%ld ", fl->fl_pid,
>>                              inode->i_sb->s_id, inode->i_ino);
>>  #else
>>              /* userspace relies on this representation of dev_t ;-( */
>> -            out += sprintf(out, "%d %02x:%02x:%ld ", fl->fl_pid,
>> +            seq_printf(f, "%d %02x:%02x:%ld ", fl->fl_pid,
>>                              MAJOR(inode->i_sb->s_dev),
>>                              MINOR(inode->i_sb->s_dev), inode->i_ino);
>>  #endif
>>      } else {
>> -            out += sprintf(out, "%d <none>:0 ", fl->fl_pid);
>> +            seq_printf(f, "%d <none>:0 ", fl->fl_pid);
>>      }
>>      if (IS_POSIX(fl)) {
>>              if (fl->fl_end == OFFSET_MAX)
>> -                    out += sprintf(out, "%Ld EOF\n", fl->fl_start);
>> +                    seq_printf(f, "%Ld EOF\n", fl->fl_start);
>>              else
>> -                    out += sprintf(out, "%Ld %Ld\n", fl->fl_start,
>> -                                    fl->fl_end);
>> +                    seq_printf(f, "%Ld %Ld\n", fl->fl_start, fl->fl_end);
>>      } else {
>> -            out += sprintf(out, "0 EOF\n");
>> +            seq_printf(f, "0 EOF\n");
>>      }
>>  }
>>  
>> -static void move_lock_status(char **p, off_t* pos, off_t offset)
>> +static int locks_show(struct seq_file *f, void *v)
>>  {
>> -    int len;
>> -    len = strlen(*p);
>> -    if(*pos >= offset) {
>> -            /* the complete line is valid */
>> -            *p += len;
>> -            *pos += len;
>> -            return;
>> -    }
>> -    if(*pos+len > offset) {
>> -            /* use the second part of the line */
>> -            int i = offset-*pos;
>> -            memmove(*p,*p+i,len-i);
>> -            *p += len-i;
>> -            *pos += len;
>> -            return;
>> -    }
>> -    /* discard the complete line */
>> -    *pos += len;
>> -}
>> +    int idx;
>> +    struct file_lock *fl, *bfl;
>>  
>> -/**
>> - *  get_locks_status        -       reports lock usage in /proc/locks
>> - *  @buffer: address in userspace to write into
>> - *  @start: ?
>> - *  @offset: how far we are through the buffer
>> - *  @length: how much to read
>> - */
>> +    fl = list_entry(v, struct file_lock, fl_link);
>> +    idx = (int)f->private;
>>  
>> -int get_locks_status(char *buffer, char **start, off_t offset, int length)
>> -{
>> -    struct file_lock *fl;
>> -    char *q = buffer;
>> -    off_t pos = 0;
>> -    int i = 0;
>> +    lock_get_status(f, fl, idx, "");
>>  
>> -    lock_kernel();
>> -    list_for_each_entry(fl, &file_lock_list, fl_link) {
>> -            struct file_lock *bfl;
>> +    list_for_each_entry(bfl, &fl->fl_block, fl_block)
>> +            lock_get_status(f, bfl, idx, " ->");
>>  
>> -            lock_get_status(q, fl, ++i, "");
>> -            move_lock_status(&q, &pos, offset);
>> +    f->private = (void *)(idx + 1);
>> +    return 0;
>> +}
>>  
>> -            if(pos >= offset+length)
>> -                    goto done;
>> +static void *locks_start(struct seq_file *f, loff_t *pos)
>> +{
>> +    lock_kernel();
>> +    f->private = (void *)1;
>> +    return seq_list_start(&file_lock_list, *pos);
>> +}
>>  
>> -            list_for_each_entry(bfl, &fl->fl_block, fl_block) {
>> -                    lock_get_status(q, bfl, i, " ->");
>> -                    move_lock_status(&q, &pos, offset);
>> +static void *locks_next(struct seq_file *f, void *v, loff_t *pos)
>> +{
>> +    return seq_list_next(v, &file_lock_list, pos);
>> +}
>>  
>> -                    if(pos >= offset+length)
>> -                            goto done;
>> -            }
>> -    }
>> -done:
>> +static void locks_stop(struct seq_file *f, void *v)
>> +{
>>      unlock_kernel();
>> -    *start = buffer;
>> -    if(q-buffer < length)
>> -            return (q-buffer);
>> -    return length;
>>  }
>>  
>> +struct seq_operations locks_seq_operations = {
>> +    .start  = locks_start,
>> +    .next   = locks_next,
>> +    .stop   = locks_stop,
>> +    .show   = locks_show,
>> +};
>> +#endif
>> +
>>  /**
>>   *  lock_may_read - checks that the region is free of locks
>>   *  @inode: the inode that is being read
>> diff --git a/fs/proc/proc_misc.c b/fs/proc/proc_misc.c
>> index 166a6db..043621c 100644
>> --- a/fs/proc/proc_misc.c
>> +++ b/fs/proc/proc_misc.c
>> @@ -68,7 +68,6 @@ extern int get_stram_list(char *);
>>  extern int get_filesystem_list(char *);
>>  extern int get_exec_domain_list(char *);
>>  extern int get_dma_list(char *);
>> -extern int get_locks_status (char *, char **, off_t, int);
>>  
>>  static int proc_calc_metrics(char *page, char **start, off_t off,
>>                               int count, int *eof, int len)
>> @@ -630,16 +629,19 @@ static int cmdline_read_proc(char *page,
>>      return proc_calc_metrics(page, start, off, count, eof, len);
>>  }
>>  
>> -static int locks_read_proc(char *page, char **start, off_t off,
>> -                             int count, int *eof, void *data)
>> +extern struct seq_operations locks_seq_operations;
>> +static int locks_open(struct inode *inode, struct file *filp)
>>  {
>> -    int len = get_locks_status(page, start, off, count);
>> -
>> -    if (len < count)
>> -            *eof = 1;
>> -    return len;
>> +    return seq_open(filp, &locks_seq_operations);
>>  }
>>  
>> +static const struct file_operations proc_locks_operations = {
>> +    .open           = locks_open,
>> +    .read           = seq_read,
>> +    .llseek         = seq_lseek,
>> +    .release        = seq_release,
>> +};
>> +
>>  static int execdomains_read_proc(char *page, char **start, off_t off,
>>                               int count, int *eof, void *data)
>>  {
>> @@ -916,7 +918,6 @@ void __init proc_misc_init(void)
>>  #endif
>>              {"filesystems", filesystems_read_proc},
>>              {"cmdline",     cmdline_read_proc},
>> -            {"locks",       locks_read_proc},
>>              {"execdomains", execdomains_read_proc},
>>              {NULL,}
>>      };
>> @@ -934,6 +935,7 @@ void __init proc_misc_init(void)
>>                      entry->proc_fops = &proc_kmsg_operations;
>>      }
>>  #endif
>> +    create_seq_entry("locks", 0, &proc_locks_operations);
>>      create_seq_entry("devices", 0, &proc_devinfo_operations);
>>      create_seq_entry("cpuinfo", 0, &proc_cpuinfo_operations);
>>  #ifdef CONFIG_BLOCK
> 

-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to