<html><head><meta name="color-scheme" content="light dark"></head><body><pre style="word-wrap: break-word; white-space: pre-wrap;">smaps: per-mapping RSS and other useful bits

Signed-off-by: Robert Love &lt;rml@novell.com&gt;

 Documentation/filesystems/proc.txt |    1 
 fs/proc/base.c                     |   61 ++++++++++++++++
 fs/proc/task_mmu.c                 |  139 ++++++++++++++++++++++++++++++++++++-
 3 files changed, 199 insertions(+), 2 deletions(-)

diff -urN linux-2.6.13/Documentation/filesystems/proc.txt linux/Documentation/filesystems/proc.txt
--- linux-2.6.13/Documentation/filesystems/proc.txt	2005-08-28 19:41:01.000000000 -0400
+++ linux/Documentation/filesystems/proc.txt	2005-08-29 13:11:55.000000000 -0400
@@ -133,6 +133,7 @@
  statm   Process memory status information              
  status  Process status in human readable form          
  wchan   If CONFIG_KALLSYMS is set, a pre-decoded wchan
+ smaps	 Extension based on maps, presenting the rss size for each mapped file
 ..............................................................................
 
 For example, to get the status information of a process, all you have to do is
diff -urN linux-2.6.13/fs/proc/base.c linux/fs/proc/base.c
--- linux-2.6.13/fs/proc/base.c	2005-08-28 19:41:01.000000000 -0400
+++ linux/fs/proc/base.c	2005-08-29 13:11:55.000000000 -0400
@@ -11,6 +11,40 @@
  *  go into icache. We cache the reference to task_struct upon lookup too.
  *  Eventually it should become a filesystem in its own. We don't use the
  *  rest of procfs anymore.
+ *
+ *
+ *  Changelog:
+ *  17-Jan-2005
+ *  Allan Bezerra
+ *  Bruna Moreira &lt;bruna.moreira@indt.org.br&gt;
+ *  Edjard Mota &lt;edjard.mota@indt.org.br&gt;
+ *  Ilias Biris &lt;ilias.biris@indt.org.br&gt;
+ *  Mauricio Lin &lt;mauricio.lin@indt.org.br&gt;
+ *
+ *  Embedded Linux Lab - 10LE Instituto Nokia de Tecnologia - INdT
+ *
+ *  A new process specific entry (smaps) included in /proc. It shows the
+ *  size of rss for each memory area. The maps entry lacks information
+ *  about physical memory size (rss) for each mapped file, i.e.,
+ *  rss information for executables and library files.
+ *  This additional information is useful for any tools that need to know
+ *  about physical memory consumption for a process specific library.
+ *
+ *  Changelog:
+ *  21-Feb-2005
+ *  Embedded Linux Lab - 10LE Instituto Nokia de Tecnologia - INdT
+ *  Pud inclusion in the page table walking.
+ *
+ *  ChangeLog:
+ *  10-Mar-2005
+ *  10LE Instituto Nokia de Tecnologia - INdT:
+ *  A better way to walks through the page table as suggested by Hugh Dickins.
+ *
+ *  Simo Piiroinen &lt;simo.piiroinen@nokia.com&gt;:
+ *  Smaps information related to shared, private, clean and dirty pages.
+ *
+ *  Paul Mundt &lt;paul.mundt@nokia.com&gt;:
+ *  Overall revision about smaps.
  */
 
 #include &lt;asm/uaccess.h&gt;
@@ -67,6 +101,7 @@
 	PROC_TGID_MAPS,
 	PROC_TGID_MOUNTS,
 	PROC_TGID_WCHAN,
+	PROC_TGID_SMAPS,
 #ifdef CONFIG_SCHEDSTATS
 	PROC_TGID_SCHEDSTAT,
 #endif
@@ -104,6 +139,7 @@
 	PROC_TID_MAPS,
 	PROC_TID_MOUNTS,
 	PROC_TID_WCHAN,
+	PROC_TID_SMAPS,
 #ifdef CONFIG_SCHEDSTATS
 	PROC_TID_SCHEDSTAT,
 #endif
@@ -152,6 +188,7 @@
 	E(PROC_TGID_ROOT,      "root",    S_IFLNK|S_IRWXUGO),
 	E(PROC_TGID_EXE,       "exe",     S_IFLNK|S_IRWXUGO),
 	E(PROC_TGID_MOUNTS,    "mounts",  S_IFREG|S_IRUGO),
+	E(PROC_TGID_SMAPS,     "smaps",   S_IFREG|S_IRUGO),
 #ifdef CONFIG_SECURITY
 	E(PROC_TGID_ATTR,      "attr",    S_IFDIR|S_IRUGO|S_IXUGO),
 #endif
@@ -188,6 +225,7 @@
 	E(PROC_TID_ROOT,       "root",    S_IFLNK|S_IRWXUGO),
 	E(PROC_TID_EXE,        "exe",     S_IFLNK|S_IRWXUGO),
 	E(PROC_TID_MOUNTS,     "mounts",  S_IFREG|S_IRUGO),
+	E(PROC_TID_SMAPS,      "smaps",   S_IFREG|S_IRUGO),
 #ifdef CONFIG_SECURITY
 	E(PROC_TID_ATTR,       "attr",    S_IFDIR|S_IRUGO|S_IXUGO),
 #endif
@@ -515,6 +553,25 @@
 	.release	= seq_release,
 };
 
+extern struct seq_operations proc_pid_smaps_op;
+static int smaps_open(struct inode *inode, struct file *file)
+{
+	struct task_struct *task = proc_task(inode);
+	int ret = seq_open(file, &amp;proc_pid_smaps_op);
+	if (!ret) {
+		struct seq_file *m = file-&gt;private_data;
+		m-&gt;private = task;
+	}
+	return ret;
+}
+
+static struct file_operations proc_smaps_operations = {
+	.open		= smaps_open,
+	.read		= seq_read,
+	.llseek		= seq_lseek,
+	.release	= seq_release,
+};
+
 extern struct seq_operations mounts_op;
 static int mounts_open(struct inode *inode, struct file *file)
 {
@@ -1539,6 +1596,10 @@
 		case PROC_TGID_MOUNTS:
 			inode-&gt;i_fop = &amp;proc_mounts_operations;
 			break;
+		case PROC_TID_SMAPS:
+		case PROC_TGID_SMAPS:
+			inode-&gt;i_fop = &amp;proc_smaps_operations;
+			break;
 #ifdef CONFIG_SECURITY
 		case PROC_TID_ATTR:
 			inode-&gt;i_nlink = 2;
diff -urN linux-2.6.13/fs/proc/task_mmu.c linux/fs/proc/task_mmu.c
--- linux-2.6.13/fs/proc/task_mmu.c	2005-08-28 19:41:01.000000000 -0400
+++ linux/fs/proc/task_mmu.c	2005-08-29 13:11:55.000000000 -0400
@@ -2,8 +2,10 @@
 #include &lt;linux/hugetlb.h&gt;
 #include &lt;linux/mount.h&gt;
 #include &lt;linux/seq_file.h&gt;
+#include &lt;linux/highmem.h&gt;
 #include &lt;asm/elf.h&gt;
 #include &lt;asm/uaccess.h&gt;
+#include &lt;asm/tlbflush.h&gt;
 #include "internal.h"
 
 char *task_mem(struct mm_struct *mm, char *buffer)
@@ -118,9 +120,9 @@
 	 * Print the dentry name for named mappings, and a
 	 * special [heap] marker for the heap:
 	 */
-	if (map-&gt;vm_file) {
+	if (map-&gt;file) {
 		pad_len_spaces(m, len);
-		seq_path(m, file-&gt;f_vfsmnt, file-&gt;f_dentry, "");
+		seq_path(m, file-&gt;f_vfsmnt, file-&gt;f_dentry, "\n");
 	} else {
 		if (mm) {
 			if (map-&gt;vm_start &lt;= mm-&gt;start_brk &amp;&amp;
@@ -146,6 +148,132 @@
 	return 0;
 }
 
+struct mem_size_stats
+{
+	unsigned long resident;
+	unsigned long shared_clean;
+	unsigned long shared_dirty;
+	unsigned long private_clean;
+	unsigned long private_dirty;
+};
+
+static void smaps_pte_range(struct vm_area_struct *vma, pmd_t *pmd,
+				unsigned long addr, unsigned long end,
+				struct mem_size_stats *mss)
+{
+	pte_t *pte, ptent;
+	unsigned long pfn;
+	struct page *page;
+
+	pte = pte_offset_map(pmd, addr);
+	do {
+		ptent = *pte;
+		if (pte_none(ptent) || !pte_present(ptent))
+			continue;
+
+		mss-&gt;resident += PAGE_SIZE;
+		pfn = pte_pfn(ptent);
+		if (!pfn_valid(pfn))
+			continue;
+
+		page = pfn_to_page(pfn);
+		if (page_count(page) &gt;= 2) {
+			if (pte_dirty(ptent))
+				mss-&gt;shared_dirty += PAGE_SIZE;
+			else
+				mss-&gt;shared_clean += PAGE_SIZE;
+		} else {
+			if (pte_dirty(ptent))
+				mss-&gt;private_dirty += PAGE_SIZE;
+			else
+				mss-&gt;private_clean += PAGE_SIZE;
+		}
+	} while (pte++, addr += PAGE_SIZE, addr != end);
+	pte_unmap(pte - 1);
+	cond_resched_lock(&amp;vma-&gt;vm_mm-&gt;page_table_lock);
+}
+
+static inline void smaps_pmd_range(struct vm_area_struct *vma, pud_t *pud,
+				unsigned long addr, unsigned long end,
+				struct mem_size_stats *mss)
+{
+	pmd_t *pmd;
+	unsigned long next;
+
+	pmd = pmd_offset(pud, addr);
+	do {
+		next = pmd_addr_end(addr, end);
+		if (pmd_none_or_clear_bad(pmd))
+			continue;
+		smaps_pte_range(vma, pmd, addr, next, mss);
+	} while (pmd++, addr = next, addr != end);
+}
+
+static inline void smaps_pud_range(struct vm_area_struct *vma, pgd_t *pgd,
+				unsigned long addr, unsigned long end,
+				struct mem_size_stats *mss)
+{
+	pud_t *pud;
+	unsigned long next;
+
+	pud = pud_offset(pgd, addr);
+	do {
+		next = pud_addr_end(addr, end);
+		if (pud_none_or_clear_bad(pud))
+			continue;
+		smaps_pmd_range(vma, pud, addr, next, mss);
+	} while (pud++, addr = next, addr != end);
+}
+
+static inline void smaps_pgd_range(struct vm_area_struct *vma,
+				unsigned long addr, unsigned long end,
+				struct mem_size_stats *mss)
+{
+	pgd_t *pgd;
+	unsigned long next;
+
+	pgd = pgd_offset(vma-&gt;vm_mm, addr);
+	do {
+		next = pgd_addr_end(addr, end);
+		if (pgd_none_or_clear_bad(pgd))
+			continue;
+		smaps_pud_range(vma, pgd, addr, next, mss);
+	} while (pgd++, addr = next, addr != end);
+}
+
+static int show_smap(struct seq_file *m, void *v)
+{
+	struct vm_area_struct *map = v;
+	struct mm_struct *mm = map-&gt;vm_mm;
+	unsigned long vma_len = (map-&gt;vm_end - map-&gt;vm_start);
+	struct mem_size_stats mss;
+
+	memset(&amp;mss, 0, sizeof mss);
+
+	show_map(m, v);
+
+	if (mm) {
+		spin_lock(&amp;mm-&gt;page_table_lock);
+		smaps_pgd_range(vma, vma-&gt;vm_start, vma-&gt;vm_end, &amp;mss);
+		spin_unlock(&amp;mm-&gt;page_table_lock);
+	}
+
+	seq_printf(m,
+		   "Size:          %8lu kB\n"
+		   "Rss:           %8lu kB\n"
+		   "Shared_Clean:  %8lu kB\n"
+		   "Shared_Dirty:  %8lu kB\n"
+		   "Private_Clean: %8lu kB\n"
+		   "Private_Dirty: %8lu kB\n",
+		   vma_len &gt;&gt; 10,
+		   mss.resident &gt;&gt; 10,
+		   mss.shared_clean  &gt;&gt; 10,
+		   mss.shared_dirty  &gt;&gt; 10,
+		   mss.private_clean &gt;&gt; 10,
+		   mss.private_dirty &gt;&gt; 10);
+	return 0;
+}
+
 static void *m_start(struct seq_file *m, loff_t *pos)
 {
 	struct task_struct *task = m-&gt;private;
@@ -233,3 +361,10 @@
 	.stop	= m_stop,
 	.show	= show_map
 };
+
+struct seq_operations proc_pid_smaps_op = {
+	.start	= m_start,
+	.next	= m_next,
+	.stop	= m_stop,
+	.show	= show_smap
+}
</pre></body></html>