ViewVC Help
View File | Revision Log | Show Annotations | Revision Graph | Root Listing
root/i-scream/projects/libstatgrab/src/libstatgrab/memory_stats.c
Revision: 1.16
Committed: Fri Oct 24 17:26:43 2003 UTC (20 years, 6 months ago) by ats
Content type: text/plain
Branch: MAIN
Changes since 1.15: +2 -3 lines
Log Message:
Support memory, swap and paging stats on Linux 2.6. For memory and swap,
this just means reading the new-style data in /proc/meminfo (since it's
present in 2.2 and 2.4 too); for paging, this means trying to read
/proc/vmstat if it's available, else reading /proc/stat (for 2.2).

File Contents

# Content
1 /*
2 * i-scream central monitoring system
3 * http://www.i-scream.org
4 * Copyright (C) 2000-2003 i-scream
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
19 */
20
21 #ifdef HAVE_CONFIG_H
22 #include "config.h"
23 #endif
24
25 #include "statgrab.h"
26 #include "tools.h"
27 #ifdef SOLARIS
28 #include <unistd.h>
29 #include <kstat.h>
30 #endif
31 #ifdef LINUX
32 #include <stdio.h>
33 #include <string.h>
34 #endif
35 #ifdef FREEBSD
36 #include <sys/types.h>
37 #include <sys/sysctl.h>
38 #include <unistd.h>
39 #endif
40
41 mem_stat_t *get_memory_stats(){
42
43 static mem_stat_t mem_stat;
44
45 #ifdef SOLARIS
46 kstat_ctl_t *kc;
47 kstat_t *ksp;
48 kstat_named_t *kn;
49 long totalmem;
50 int pagesize;
51 #endif
52 #ifdef LINUX
53 char *line_ptr;
54 unsigned long long value;
55 FILE *f;
56 #endif
57 #ifdef FREEBSD
58 int mib[2];
59 u_long physmem;
60 size_t size;
61 u_int free_count;
62 u_int cache_count;
63 u_int inactive_count;
64 int pagesize;
65 #endif
66 #ifdef NETBSD
67 struct uvmexp *uvm;
68 #endif
69
70 #ifdef SOLARIS
71 if((pagesize=sysconf(_SC_PAGESIZE)) == -1){
72 return NULL;
73 }
74
75 if((totalmem=sysconf(_SC_PHYS_PAGES)) == -1){
76 return NULL;
77 }
78
79 if ((kc = kstat_open()) == NULL) {
80 return NULL;
81 }
82 if((ksp=kstat_lookup(kc, "unix", 0, "system_pages")) == NULL){
83 return NULL;
84 }
85 if (kstat_read(kc, ksp, 0) == -1) {
86 return NULL;
87 }
88 if((kn=kstat_data_lookup(ksp, "freemem")) == NULL){
89 return NULL;
90 }
91 kstat_close(kc);
92
93 mem_stat.total = (long long)totalmem * (long long)pagesize;
94 mem_stat.free = ((long long)kn->value.ul) * (long long)pagesize;
95 mem_stat.used = mem_stat.total - mem_stat.free;
96 #endif
97
98 #ifdef LINUX
99 if ((f = fopen("/proc/meminfo", "r")) == NULL) {
100 return NULL;
101 }
102
103 while ((line_ptr = f_read_line(f, "")) != NULL) {
104 if (sscanf(line_ptr, "%*s %llu kB", &value) != 1) {
105 continue;
106 }
107 value *= 1024;
108
109 if (strncmp(line_ptr, "MemTotal:", 9) == 0) {
110 mem_stat.total = value;
111 } else if (strncmp(line_ptr, "MemFree:", 8) == 0) {
112 mem_stat.free = value;
113 } else if (strncmp(line_ptr, "Cached:", 7) == 0) {
114 mem_stat.cache = value;
115 }
116 }
117
118 fclose(f);
119 mem_stat.used = mem_stat.total - mem_stat.free;
120 #endif
121
122 #ifdef FREEBSD
123 /* Returns bytes */
124 mib[0] = CTL_HW;
125 mib[1] = HW_PHYSMEM;
126 size = sizeof physmem;
127 if (sysctl(mib, 2, &physmem, &size, NULL, 0) < 0) {
128 return NULL;
129 }
130 mem_stat.total = physmem;
131
132 /*returns pages*/
133 size = sizeof free_count;
134 if (sysctlbyname("vm.stats.vm.v_free_count", &free_count, &size, NULL, 0) < 0){
135 return NULL;
136 }
137
138 size = sizeof inactive_count;
139 if (sysctlbyname("vm.stats.vm.v_inactive_count", &inactive_count , &size, NULL, 0) < 0){
140 return NULL;
141 }
142
143 size = sizeof cache_count;
144 if (sysctlbyname("vm.stats.vm.v_cache_count", &cache_count, &size, NULL, 0) < 0){
145 return NULL;
146 }
147
148 /* Because all the vm.stats returns pages, I need to get the page size.
149 * After that I then need to multiple the anything that used vm.stats to
150 * get the system statistics by pagesize
151 */
152 if ((pagesize=getpagesize()) == -1){
153 return NULL;
154 }
155
156 mem_stat.cache=cache_count*pagesize;
157
158 /* Of couse nothing is ever that simple :) And I have inactive pages to
159 * deal with too. So I'm going to add them to free memory :)
160 */
161 mem_stat.free=(free_count*pagesize)+(inactive_count*pagesize);
162 mem_stat.used=physmem-mem_stat.free;
163 #endif
164
165 #ifdef NETBSD
166 if ((uvm = get_uvmexp()) == NULL) {
167 return NULL;
168 }
169 mem_stat.total = uvm->pagesize * uvm->npages;
170 mem_stat.cache = uvm->pagesize * (uvm->filepages + uvm->execpages);
171 mem_stat.free = uvm->pagesize * (uvm->free + uvm->inactive);
172 mem_stat.used = mem_stat.total - mem_stat.free;
173 #endif
174
175 return &mem_stat;
176 }