On Thu, May 28, 2015 at 01:14:35AM +0300, Cyrill Gorcunov wrote:
> On Thu, May 28, 2015 at 12:49:53AM +0300, Alexey Dobriyan wrote:
> ...
> > -static int proc_pid_cmdline(struct seq_file *m, struct pid_namespace *ns,
> > -                       struct pid *pid, struct task_struct *task)
> > +static ssize_t proc_pid_cmdline_read(struct file *file, char __user *buf,
> > +                                size_t _count, loff_t *pos)
> >  {
> > +   struct task_struct *tsk;
> > +   struct mm_struct *mm;
> > +   char *page;
> > +   unsigned long count = _count;
> > +   unsigned long arg_start, arg_end, env_start, env_end;
> > +   unsigned long len1, len2, len;
> > +   unsigned long p;
> > +   char c;
> > +   ssize_t rv;
> > +
> > +   BUG_ON(*pos < 0);
> > +
> > +   tsk = get_proc_task(file_inode(file));
> > +   if (!tsk)
> > +           return -ESRCH;
> > +   mm = get_task_mm(tsk);
> > +   put_task_struct(tsk);
> > +   if (!mm)
> > +           return 0;
> > +   /* Check if process spawned far enough to have cmdline. */
> > +   if (!mm->env_end) {
> > +           rv = 0;
> > +           goto out_mmput;
> > +   }
> > +
> > +   page = (char *)__get_free_page(GFP_TEMPORARY);
> > +   if (!page) {
> > +           rv = -ENOMEM;
> > +           goto out_mmput;
> > +   }
> > +
> > +   down_read(&mm->mmap_sem);
> > +   arg_start = mm->arg_start;
> > +   arg_end = mm->arg_end;
> > +   env_start = mm->env_start;
> > +   env_end = mm->env_end;
> > +   up_read(&mm->mmap_sem);
> 
> Could you please explain why this down/up is needed?

Code is written this way to get constistent snapshot of data.
If you look at PR_SET_MM_* code, you'll notice down_read(&mm->mmap_sem)
as well which is a separate bug because you're _writing_ those fields
eventually in prctl_set_mm(), yuck!
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
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