// Process memory query and utility functions. // Copyright (C) 2009, 2010 Red Hat Inc. // // This file is part of systemtap, and is free software. You can // redistribute it and/or modify it under the terms of the GNU General // Public License (GPL); either version 2, or (at your option) any // later version. // // Process memory query and utility functions provide information about // the memory usage of the current application. These functions provide // information about the full size, resident, shared, code and data used // by the current process. And provide utility functions to query the // page size of the current architecture and create human readable string // representations of bytes and pages used. // %{ /* PF_BORROWED_MM got renamed to PF_KTHREAD with same semantics somewhere. */ #ifdef PF_BORROWED_MM #define _STP_PF_KTHREAD PF_BORROWED_MM #else #define _STP_PF_KTHREAD PF_KTHREAD #endif /* Returns the mm for the current proc. Slightly paranoid. Only returns if the task isn't starting, exiting or (coopted by) a kernel thread. */ static struct mm_struct *_stp_proc_mm(void) { if (current->flags & (_STP_PF_KTHREAD | PF_EXITING | PF_STARTING)) return NULL; return current->mm; } %} function _stp_get_mm_counter_file_rss:long(mm:long) { %( CONFIG_NR_CPUS >= CONFIG_SPLIT_PTLOCK_CPUS %? return @cast(mm, "mm_struct", "kernel")->_file_rss->counter; %: return @cast(mm, "mm_struct", "kernel")->_file_rss; %) } function _stp_get_mm_counter_anon_rss(mm:long) { %( CONFIG_NR_CPUS >= CONFIG_SPLIT_PTLOCK_CPUS %? return @cast(mm, "mm_struct", "kernel")->_anon_rss->counter; %: return @cast(mm, "mm_struct", "kernel")->_anon_rss; %) } /** * sfunction proc_mem_size - Total program virtual memory size in pages * * Description: Returns the total virtual memory size in pages of the * current process, or zero when there is no current process or the * number of pages couldn't be retrieved. */ function proc_mem_size:long () %{ /* pure */ /* unprivileged */ struct mm_struct *mm = _stp_proc_mm (); if (mm) THIS->__retvalue = mm->total_vm; else THIS->__retvalue = 0; %} /** * sfunction proc_mem_size_pid - Total program virtual memory size in pages * * @pid: The pid of process to examine * * Description: Returns the total virtual memory size in pages of the * given process, or zero when that process doesn't exist or the * number of pages couldn't be retrieved. */ function proc_mem_size_pid:long (pid:long) { task = pid2task(pid); if (task != 0) { mm = @cast(task, "task_struct", "kernel")->mm; if (mm != 0) return @cast(mm, "mm_struct", "kernel")->total_vm; } return 0; } /** * sfunction proc_mem_rss - Program resident set size in pages * * Description: Returns the resident set size in pages of the current * process, or zero when there is no current process or the number of * pages couldn't be retrieved. */ function proc_mem_rss:long () %{ /* pure */ /* unprivileged */ struct mm_struct *mm = _stp_proc_mm (); if (mm) THIS->__retvalue = (get_mm_counter(mm, file_rss) + get_mm_counter(mm, anon_rss)); else THIS->__retvalue = 0; %} /** * sfunction proc_mem_rss_pid - Program resident set size in pages * * @pid: The pid of process to examine * * Description: Returns the resident set size in pages of the given * process, or zero when the process doesn't exist or the number of * pages couldn't be retrieved. */ function proc_mem_rss_pid:long (pid:long) { task = pid2task(pid); if (task) { mm = @cast(task, "task_struct", "kernel")->mm; if (mm != 0) return (_stp_get_mm_counter_file_rss (mm) + _stp_get_mm_counter_anon_rss (mm)); } return 0; } /** * sfunction proc_mem_shr - Program shared pages (from shared mappings) * * Description: Returns the shared pages (from shared mappings) of the * current process, or zero when there is no current process or the * number of pages couldn't be retrieved. */ function proc_mem_shr:long () %{ /* pure */ /* unprivileged */ struct mm_struct *mm = _stp_proc_mm (); if (mm) THIS->__retvalue = get_mm_counter(mm, file_rss); else THIS->__retvalue = 0; %} /** * sfunction proc_mem_shr_pid - Program shared pages (from shared mappings) * * @pid: The pid of process to examine * * Description: Returns the shared pages (from shared mappings) of the * given process, or zero when the process doesn't exist or the * number of pages couldn't be retrieved. */ function proc_mem_shr_pid:long (pid:long) { task = pid2task(pid); if (task) { mm = @cast(task, "task_struct", "kernel")->mm; if (mm != 0) return _stp_get_mm_counter_file_rss (mm); } return 0; } /** * sfunction proc_mem_txt - Program text (code) size in pages * * Description: Returns the current process text (code) size in pages, * or zero when there is no current process or the number of pages * couldn't be retrieved. */ function proc_mem_txt:long () %{ /* pure */ /* unprivileged */ struct mm_struct *mm = _stp_proc_mm (); if (mm) THIS->__retvalue = (PAGE_ALIGN(mm->end_code) - (mm->start_code & PAGE_MASK)) >> PAGE_SHIFT; else THIS->__retvalue = 0; %} function _stp_mem_txt_adjust:long (start_code:long, end_code:long) %{ /* pure */ unsigned long start_code = (unsigned long) THIS->start_code; unsigned long end_code = (unsigned long) THIS->end_code; THIS->__retvalue = (PAGE_ALIGN(end_code) - (start_code & PAGE_MASK)) >> PAGE_SHIFT; %} /** * sfunction proc_mem_txt_pid - Program text (code) size in pages * * @pid: The pid of process to examine * * Description: Returns the given process text (code) size in pages, * or zero when the process doesn't exist or the number of pages * couldn't be retrieved. */ function proc_mem_txt_pid:long (pid:long) { task = pid2task(pid); if (task != 0) { mm = @cast(task, "task_struct", "kernel")->mm; if (mm != 0) { s = @cast(mm, "mm_struct", "kernel")->start_code; e = @cast(mm, "mm_struct", "kernel")->end_code; return _stp_mem_txt_adjust (s, e); } } return 0; } /** * sfunction proc_mem_data - Program data size (data + stack) in pages * * Description: Returns the current process data size (data + stack) * in pages, or zero when there is no current process or the number of * pages couldn't be retrieved. */ function proc_mem_data:long () %{ /* pure */ /* unprivileged */ struct mm_struct *mm = _stp_proc_mm (); if (mm) THIS->__retvalue = mm->total_vm - mm->shared_vm; else THIS->__retvalue = 0; %} /** * sfunction proc_mem_data_pid - Program data size (data + stack) in pages * * @pid: The pid of process to examine * * Description: Returns the given process data size (data + stack) * in pages, or zero when the process doesn't exist or the number of * pages couldn't be retrieved. */ function proc_mem_data_pid:long (pid:long) { task = pid2task(pid); if (task != 0) { mm = @cast(task, "task_struct", "kernel")->mm; if (mm != 0) { t = @cast(mm, "mm_struct", "kernel")->total_vm; s = @cast(mm, "mm_struct", "kernel")->shared_vm; return t - s; } } return 0; } /** * sfunction mem_page_size - Number of bytes in a page for this architecture */ function mem_page_size:long () %{ /* pure */ /* unprivileged */ THIS->__retvalue = PAGE_SIZE; %} // Return a 5 character wide string " x.yyp", " xx.yp", " xxxp", "xxxxp". function _stp_number_to_string_postfix:string (x:long, y:long, p:string) { if (x < 10) return sprintf("%d.%.2d%s", x, y * 100 / 1024, p); if (x < 100) return sprintf("%2d.%d%s", x, y * 10 / 1024, p); return sprintf("%4d%s", x, p); } /** * sfunction bytes_to_string - Human readable string for given bytes * * @bytes: Number of bytes to translate. * * Description: Returns a string representing the number of bytes (up * to 1024 bytes), the number of kilobytes (when less than 1024K) * postfixed by 'K', the number of megabytes (when less than 1024M) * postfixed by 'M' or the number of gigabytes postfixed by 'G'. If * representing K, M or G, and the number is amount is less than 100, * it includes a '.' plus the remainer. The returned string will be 5 * characters wide (padding with whitespace at the front) unless * negative or representing more than 9999G bytes. */ function bytes_to_string:string (bytes:long) { if (bytes < 1024) return sprintf("%5d", bytes); remain = bytes % 1024; bytes = bytes / 1024; if (bytes < 1024) return _stp_number_to_string_postfix(bytes, remain, "K"); remain = bytes % 1024; bytes = bytes / 1024; if (bytes < 1024) return _stp_number_to_string_postfix(bytes, remain, "M"); remain = bytes % 1024; bytes = bytes / 1024; return _stp_number_to_string_postfix(bytes, remain, "G"); } /** * sfunction pages_to_string - Turns pages into a human readable string * * @pages: Number of pages to translate. * * Description: Multiplies pages by page_size() to get the number of * bytes and returns the result of bytes_to_string(). */ function pages_to_string:string (pages:long) { bytes = pages * mem_page_size(); return bytes_to_string (bytes); } /** * sfunction proc_mem_string - Human readable string of current proc memory usage * * Description: Returns a human readable string showing the size, rss, * shr, txt and data of the memory used by the current process. * For example "size: 301m, rss: 11m, shr: 8m, txt: 52k, data: 2248k". */ function proc_mem_string:string () { return sprintf ("size: %s, rss: %s, shr: %s, txt: %s, data: %s", pages_to_string(proc_mem_size()), pages_to_string(proc_mem_rss()), pages_to_string(proc_mem_shr()), pages_to_string(proc_mem_txt()), pages_to_string(proc_mem_data())); } /** * sfunction proc_mem_string_pid - Human readable string of process memory usage * * @pid: The pid of process to examine * * Description: Returns a human readable string showing the size, rss, * shr, txt and data of the memory used by the given process. * For example "size: 301m, rss: 11m, shr: 8m, txt: 52k, data: 2248k". */ function proc_mem_string_pid:string (pid:long) { return sprintf ("size: %s, rss: %s, shr: %s, txt: %s, data: %s", pages_to_string(proc_mem_size_pid(pid)), pages_to_string(proc_mem_rss_pid(pid)), pages_to_string(proc_mem_shr_pid(pid)), pages_to_string(proc_mem_txt_pid(pid)), pages_to_string(proc_mem_data_pid(pid))); }