diff options
Diffstat (limited to 'fs/proc/array.c')
-rw-r--r-- | fs/proc/array.c | 85 |
1 files changed, 84 insertions, 1 deletions
diff --git a/fs/proc/array.c b/fs/proc/array.c index 725a650bbbb8..0c6bc602e6c4 100644 --- a/fs/proc/array.c +++ b/fs/proc/array.c | |||
@@ -82,6 +82,7 @@ | |||
82 | #include <linux/pid_namespace.h> | 82 | #include <linux/pid_namespace.h> |
83 | #include <linux/ptrace.h> | 83 | #include <linux/ptrace.h> |
84 | #include <linux/tracehook.h> | 84 | #include <linux/tracehook.h> |
85 | #include <linux/swapops.h> | ||
85 | 86 | ||
86 | #include <asm/pgtable.h> | 87 | #include <asm/pgtable.h> |
87 | #include <asm/processor.h> | 88 | #include <asm/processor.h> |
@@ -321,6 +322,87 @@ static inline void task_context_switch_counts(struct seq_file *m, | |||
321 | p->nivcsw); | 322 | p->nivcsw); |
322 | } | 323 | } |
323 | 324 | ||
325 | struct stack_stats { | ||
326 | struct vm_area_struct *vma; | ||
327 | unsigned long startpage; | ||
328 | unsigned long usage; | ||
329 | }; | ||
330 | |||
331 | static int stack_usage_pte_range(pmd_t *pmd, unsigned long addr, | ||
332 | unsigned long end, struct mm_walk *walk) | ||
333 | { | ||
334 | struct stack_stats *ss = walk->private; | ||
335 | struct vm_area_struct *vma = ss->vma; | ||
336 | pte_t *pte, ptent; | ||
337 | spinlock_t *ptl; | ||
338 | int ret = 0; | ||
339 | |||
340 | pte = pte_offset_map_lock(vma->vm_mm, pmd, addr, &ptl); | ||
341 | for (; addr != end; pte++, addr += PAGE_SIZE) { | ||
342 | ptent = *pte; | ||
343 | |||
344 | #ifdef CONFIG_STACK_GROWSUP | ||
345 | if (pte_present(ptent) || is_swap_pte(ptent)) | ||
346 | ss->usage = addr - ss->startpage + PAGE_SIZE; | ||
347 | #else | ||
348 | if (pte_present(ptent) || is_swap_pte(ptent)) { | ||
349 | ss->usage = ss->startpage - addr + PAGE_SIZE; | ||
350 | pte++; | ||
351 | ret = 1; | ||
352 | break; | ||
353 | } | ||
354 | #endif | ||
355 | } | ||
356 | pte_unmap_unlock(pte - 1, ptl); | ||
357 | cond_resched(); | ||
358 | return ret; | ||
359 | } | ||
360 | |||
361 | static inline unsigned long get_stack_usage_in_bytes(struct vm_area_struct *vma, | ||
362 | struct task_struct *task) | ||
363 | { | ||
364 | struct stack_stats ss; | ||
365 | struct mm_walk stack_walk = { | ||
366 | .pmd_entry = stack_usage_pte_range, | ||
367 | .mm = vma->vm_mm, | ||
368 | .private = &ss, | ||
369 | }; | ||
370 | |||
371 | if (!vma->vm_mm || is_vm_hugetlb_page(vma)) | ||
372 | return 0; | ||
373 | |||
374 | ss.vma = vma; | ||
375 | ss.startpage = task->stack_start & PAGE_MASK; | ||
376 | ss.usage = 0; | ||
377 | |||
378 | #ifdef CONFIG_STACK_GROWSUP | ||
379 | walk_page_range(KSTK_ESP(task) & PAGE_MASK, vma->vm_end, | ||
380 | &stack_walk); | ||
381 | #else | ||
382 | walk_page_range(vma->vm_start, (KSTK_ESP(task) & PAGE_MASK) + PAGE_SIZE, | ||
383 | &stack_walk); | ||
384 | #endif | ||
385 | return ss.usage; | ||
386 | } | ||
387 | |||
388 | static inline void task_show_stack_usage(struct seq_file *m, | ||
389 | struct task_struct *task) | ||
390 | { | ||
391 | struct vm_area_struct *vma; | ||
392 | struct mm_struct *mm = get_task_mm(task); | ||
393 | |||
394 | if (mm) { | ||
395 | down_read(&mm->mmap_sem); | ||
396 | vma = find_vma(mm, task->stack_start); | ||
397 | if (vma) | ||
398 | seq_printf(m, "Stack usage:\t%lu kB\n", | ||
399 | get_stack_usage_in_bytes(vma, task) >> 10); | ||
400 | |||
401 | up_read(&mm->mmap_sem); | ||
402 | mmput(mm); | ||
403 | } | ||
404 | } | ||
405 | |||
324 | int proc_pid_status(struct seq_file *m, struct pid_namespace *ns, | 406 | int proc_pid_status(struct seq_file *m, struct pid_namespace *ns, |
325 | struct pid *pid, struct task_struct *task) | 407 | struct pid *pid, struct task_struct *task) |
326 | { | 408 | { |
@@ -340,6 +422,7 @@ int proc_pid_status(struct seq_file *m, struct pid_namespace *ns, | |||
340 | task_show_regs(m, task); | 422 | task_show_regs(m, task); |
341 | #endif | 423 | #endif |
342 | task_context_switch_counts(m, task); | 424 | task_context_switch_counts(m, task); |
425 | task_show_stack_usage(m, task); | ||
343 | return 0; | 426 | return 0; |
344 | } | 427 | } |
345 | 428 | ||
@@ -481,7 +564,7 @@ static int do_task_stat(struct seq_file *m, struct pid_namespace *ns, | |||
481 | rsslim, | 564 | rsslim, |
482 | mm ? mm->start_code : 0, | 565 | mm ? mm->start_code : 0, |
483 | mm ? mm->end_code : 0, | 566 | mm ? mm->end_code : 0, |
484 | (permitted && mm) ? mm->start_stack : 0, | 567 | (permitted) ? task->stack_start : 0, |
485 | esp, | 568 | esp, |
486 | eip, | 569 | eip, |
487 | /* The signal information here is obsolete. | 570 | /* The signal information here is obsolete. |