Skip to content

Commit

Permalink
proc: nommu: fix empty /proc/<pid>/maps
Browse files Browse the repository at this point in the history
[ Upstream commit fe44198 ]

On no-MMU, /proc/<pid>/maps reads as an empty file.  This happens because
find_vma(mm, 0) always returns NULL (assuming no vma actually contains the
zero address, which is normally the case).

To fix this bug and improve the maintainability in the future, this patch
makes the no-MMU implementation as similar as possible to the MMU
implementation.

The only remaining differences are the lack of hold/release_task_mempolicy
and the extra code to shoehorn the gate vma into the iterator.

This has been tested on top of 6.5.3 on an STM32F746.

Link: https://lkml.kernel.org/r/[email protected]
Fixes: 0c563f1 ("proc: remove VMA rbtree use from nommu")
Signed-off-by: Ben Wolsieffer <[email protected]>
Cc: Davidlohr Bueso <[email protected]>
Cc: Giulio Benetti <[email protected]>
Cc: Liam R. Howlett <[email protected]>
Cc: Matthew Wilcox (Oracle) <[email protected]>
Cc: Oleg Nesterov <[email protected]>
Cc: Vlastimil Babka <[email protected]>
Signed-off-by: Andrew Morton <[email protected]>
Signed-off-by: Sasha Levin <[email protected]>
  • Loading branch information
lopsided98 authored and gregkh committed Oct 6, 2023
1 parent 1d45e6d commit 7e4f49c
Show file tree
Hide file tree
Showing 2 changed files with 22 additions and 17 deletions.
2 changes: 0 additions & 2 deletions fs/proc/internal.h
Original file line number Diff line number Diff line change
Expand Up @@ -289,9 +289,7 @@ struct proc_maps_private {
struct inode *inode;
struct task_struct *task;
struct mm_struct *mm;
#ifdef CONFIG_MMU
struct vma_iterator iter;
#endif
#ifdef CONFIG_NUMA
struct mempolicy *task_mempolicy;
#endif
Expand Down
37 changes: 22 additions & 15 deletions fs/proc/task_nommu.c
Original file line number Diff line number Diff line change
Expand Up @@ -188,15 +188,28 @@ static int show_map(struct seq_file *m, void *_p)
return nommu_vma_show(m, _p);
}

static void *m_start(struct seq_file *m, loff_t *pos)
static struct vm_area_struct *proc_get_vma(struct proc_maps_private *priv,
loff_t *ppos)
{
struct vm_area_struct *vma = vma_next(&priv->iter);

if (vma) {
*ppos = vma->vm_start;
} else {
*ppos = -1UL;
}

return vma;
}

static void *m_start(struct seq_file *m, loff_t *ppos)
{
struct proc_maps_private *priv = m->private;
unsigned long last_addr = *ppos;
struct mm_struct *mm;
struct vm_area_struct *vma;
unsigned long addr = *pos;

/* See m_next(). Zero at the start or after lseek. */
if (addr == -1UL)
/* See proc_get_vma(). Zero at the start or after lseek. */
if (last_addr == -1UL)
return NULL;

/* pin the task and mm whilst we play with them */
Expand All @@ -218,12 +231,9 @@ static void *m_start(struct seq_file *m, loff_t *pos)
return ERR_PTR(-EINTR);
}

/* start the next element from addr */
vma = find_vma(mm, addr);
if (vma)
return vma;
vma_iter_init(&priv->iter, mm, last_addr);

return NULL;
return proc_get_vma(priv, ppos);
}

static void m_stop(struct seq_file *m, void *v)
Expand All @@ -240,12 +250,9 @@ static void m_stop(struct seq_file *m, void *v)
priv->task = NULL;
}

static void *m_next(struct seq_file *m, void *_p, loff_t *pos)
static void *m_next(struct seq_file *m, void *_p, loff_t *ppos)
{
struct vm_area_struct *vma = _p;

*pos = vma->vm_end;
return find_vma(vma->vm_mm, vma->vm_end);
return proc_get_vma(m->private, ppos);
}

static const struct seq_operations proc_pid_maps_ops = {
Expand Down

0 comments on commit 7e4f49c

Please sign in to comment.