sys-proctable 1.0.0-universal-darwin → 1.1.0-universal-darwin
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- checksums.yaml.gz.sig +0 -0
- data.tar.gz.sig +0 -0
- data/CHANGES +11 -1
- data/MANIFEST +1 -2
- data/README +8 -16
- data/Rakefile +7 -10
- data/lib/darwin/sys/proctable.rb +241 -0
- data/lib/sys/proctable/version.rb +1 -1
- data/sys-proctable.gemspec +2 -2
- data/test/test_sys_proctable_all.rb +30 -19
- data/test/test_sys_proctable_darwin.rb +26 -180
- data/test/test_sys_top.rb +30 -11
- metadata +21 -8
- metadata.gz.sig +0 -0
- data/ext/darwin/extconf.rb +0 -4
- data/ext/darwin/sys/proctable.c +0 -451
data/ext/darwin/sys/proctable.c
DELETED
@@ -1,451 +0,0 @@
|
|
1
|
-
/**********************************************************************
|
2
|
-
* Mac OS X code for sys-proctable Ruby library. *
|
3
|
-
* *
|
4
|
-
* Date: 3-Mar-2006 (original submission date) *
|
5
|
-
* Author: David Felstead (david.felstead at gmail dot com) *
|
6
|
-
* Based on bsd.c by Daniel J. Berger (djberg96 at gmail dot com) *
|
7
|
-
*********************************************************************/
|
8
|
-
#include "ruby.h"
|
9
|
-
#include <sys/param.h>
|
10
|
-
#include <sys/stat.h>
|
11
|
-
#include <sys/sysctl.h>
|
12
|
-
#include <sys/types.h>
|
13
|
-
#include <sys/user.h>
|
14
|
-
#include <errno.h>
|
15
|
-
|
16
|
-
#include <stdio.h>
|
17
|
-
#include <stdlib.h>
|
18
|
-
#include <string.h>
|
19
|
-
#include <unistd.h>
|
20
|
-
|
21
|
-
#define pid_of(pproc) pproc->kp_proc.p_pid
|
22
|
-
|
23
|
-
#define PROC_MIB_LEN 4
|
24
|
-
#define ARGS_MIB_LEN 3
|
25
|
-
|
26
|
-
#ifndef ARGS_MAX_LEN
|
27
|
-
#define ARGS_MAX_LEN sysconf(_SC_ARG_MAX)
|
28
|
-
#endif
|
29
|
-
|
30
|
-
VALUE cProcTableError, sProcStruct;
|
31
|
-
|
32
|
-
int argv_of_pid(int pid, VALUE* v_cmdline, VALUE* v_exe, VALUE* v_environ) {
|
33
|
-
int mib[3], argmax, nargs, c = 0;
|
34
|
-
size_t size;
|
35
|
-
char *procargs, *sp, *np, *cp;
|
36
|
-
int show_args = 1;
|
37
|
-
|
38
|
-
mib[0] = CTL_KERN;
|
39
|
-
mib[1] = KERN_ARGMAX;
|
40
|
-
|
41
|
-
size = sizeof(argmax);
|
42
|
-
|
43
|
-
if (sysctl(mib, 2, &argmax, &size, NULL, 0) == -1)
|
44
|
-
goto ERROR_A;
|
45
|
-
|
46
|
-
// Allocate space for the arguments.
|
47
|
-
procargs = (char *)ruby_xmalloc(argmax);
|
48
|
-
|
49
|
-
/*
|
50
|
-
* Make a sysctl() call to get the raw argument space of the process.
|
51
|
-
* The layout is documented in start.s, which is part of the Csu
|
52
|
-
* project. In summary, it looks like:
|
53
|
-
*
|
54
|
-
* /---------------\ 0x00000000
|
55
|
-
* : :
|
56
|
-
* : :
|
57
|
-
* |---------------|
|
58
|
-
* | argc |
|
59
|
-
* |---------------|
|
60
|
-
* | arg[0] |
|
61
|
-
* |---------------|
|
62
|
-
* : :
|
63
|
-
* : :
|
64
|
-
* |---------------|
|
65
|
-
* | arg[argc - 1] |
|
66
|
-
* |---------------|
|
67
|
-
* | 0 |
|
68
|
-
* |---------------|
|
69
|
-
* | env[0] |
|
70
|
-
* |---------------|
|
71
|
-
* : :
|
72
|
-
* : :
|
73
|
-
* |---------------|
|
74
|
-
* | env[n] |
|
75
|
-
* |---------------|
|
76
|
-
* | 0 |
|
77
|
-
* |---------------| <-- Beginning of data returned by sysctl() is here.
|
78
|
-
* | argc |
|
79
|
-
* |---------------|
|
80
|
-
* | exec_path |
|
81
|
-
* |:::::::::::::::|
|
82
|
-
* | |
|
83
|
-
* | String area. |
|
84
|
-
* | |
|
85
|
-
* |---------------| <-- Top of stack.
|
86
|
-
* : :
|
87
|
-
* : :
|
88
|
-
* \---------------/ 0xffffffff
|
89
|
-
*/
|
90
|
-
mib[0] = CTL_KERN;
|
91
|
-
mib[1] = KERN_PROCARGS2;
|
92
|
-
mib[2] = pid;
|
93
|
-
|
94
|
-
size = (size_t)argmax;
|
95
|
-
|
96
|
-
if (sysctl(mib, 3, procargs, &size, NULL, 0) == -1)
|
97
|
-
goto ERROR_B;
|
98
|
-
|
99
|
-
memcpy(&nargs, procargs, sizeof(nargs));
|
100
|
-
cp = procargs + sizeof(nargs);
|
101
|
-
|
102
|
-
// Copy exec_path to ruby String.
|
103
|
-
*v_exe = rb_str_new2(cp);
|
104
|
-
|
105
|
-
// Skip the saved exec_path.
|
106
|
-
for (; cp < &procargs[size]; cp++) {
|
107
|
-
if (*cp == '\0')
|
108
|
-
break; // End of exec_path reached.
|
109
|
-
}
|
110
|
-
|
111
|
-
if (cp == &procargs[size])
|
112
|
-
goto ERROR_B;
|
113
|
-
|
114
|
-
// Skip trailing '\0' characters.
|
115
|
-
for (; cp < &procargs[size]; cp++){
|
116
|
-
if (*cp != '\0')
|
117
|
-
break; // Beginning of first argument reached.
|
118
|
-
}
|
119
|
-
|
120
|
-
if (cp == &procargs[size])
|
121
|
-
goto ERROR_B;
|
122
|
-
|
123
|
-
// Save where the argv[0] string starts.
|
124
|
-
sp = cp;
|
125
|
-
|
126
|
-
/*
|
127
|
-
* Iterate through the '\0'-terminated strings and convert '\0' to ' '
|
128
|
-
* until a string is found that has a '=' character in it (or there are
|
129
|
-
* no more strings in procargs). There is no way to deterministically
|
130
|
-
* know where the command arguments end and the environment strings
|
131
|
-
* start, which is why the '=' character is searched for as a heuristic.
|
132
|
-
*/
|
133
|
-
for (np = NULL; c < nargs && cp < &procargs[size]; cp++) {
|
134
|
-
if (*cp == '\0') {
|
135
|
-
c++;
|
136
|
-
|
137
|
-
if (np != NULL)
|
138
|
-
*np = ' '; // Convert previous '\0'.
|
139
|
-
|
140
|
-
// Note location of current '\0'.
|
141
|
-
np = cp;
|
142
|
-
|
143
|
-
if (!show_args) {
|
144
|
-
/*
|
145
|
-
* Don't convert '\0' characters to ' '.
|
146
|
-
* However, we needed to know that the
|
147
|
-
* command name was terminated, which we
|
148
|
-
* now know.
|
149
|
-
*/
|
150
|
-
break;
|
151
|
-
}
|
152
|
-
}
|
153
|
-
}
|
154
|
-
|
155
|
-
/*
|
156
|
-
* sp points to the beginning of the arguments/environment string, and
|
157
|
-
* np should point to the '\0' terminator for the string.
|
158
|
-
*/
|
159
|
-
if (np == NULL || np == sp)
|
160
|
-
goto ERROR_B; // Empty or unterminated string.
|
161
|
-
|
162
|
-
// Make a copy of the string to ruby String.
|
163
|
-
*v_cmdline = rb_str_new2(sp);
|
164
|
-
|
165
|
-
// Read environment variables to ruby Hash.
|
166
|
-
*v_environ = rb_hash_new();
|
167
|
-
|
168
|
-
while (cp[0]) {
|
169
|
-
sp = strsep(&cp, "=");
|
170
|
-
|
171
|
-
if (!sp || !cp)
|
172
|
-
break;
|
173
|
-
|
174
|
-
rb_hash_aset(*v_environ, rb_str_new2(sp), rb_str_new2(cp));
|
175
|
-
cp += strlen(cp) + 1;
|
176
|
-
}
|
177
|
-
|
178
|
-
// Cleanup.
|
179
|
-
ruby_xfree(procargs);
|
180
|
-
return 0;
|
181
|
-
|
182
|
-
ERROR_B:
|
183
|
-
ruby_xfree(procargs);
|
184
|
-
ERROR_A:
|
185
|
-
return -1;
|
186
|
-
}
|
187
|
-
|
188
|
-
/*
|
189
|
-
* call-seq:
|
190
|
-
* ProcTable.ps(pid=nil)
|
191
|
-
* ProcTable.ps(pid=nil){ |ps| ... }
|
192
|
-
*
|
193
|
-
* In block form, yields a ProcTableStruct for each process entry that you
|
194
|
-
* have rights to. This method returns an array of ProcTableStruct's in
|
195
|
-
* non-block form.
|
196
|
-
*
|
197
|
-
* If a +pid+ is provided, then only a single ProcTableStruct is yielded or
|
198
|
-
* returned, or nil if no process information is found for that +pid+.
|
199
|
-
*/
|
200
|
-
static VALUE pt_ps(int argc, VALUE* argv, VALUE klass){
|
201
|
-
int err;
|
202
|
-
char state[8];
|
203
|
-
struct kinfo_proc* procs;
|
204
|
-
VALUE v_pid, v_tty_num, v_tty_dev, v_start_time;
|
205
|
-
VALUE v_pstruct = Qnil;
|
206
|
-
VALUE v_array = rb_ary_new();
|
207
|
-
size_t length, count;
|
208
|
-
size_t i = 0;
|
209
|
-
int g;
|
210
|
-
VALUE v_cmdline, v_exe, v_environ, v_groups;
|
211
|
-
|
212
|
-
// Passed into sysctl call
|
213
|
-
static const int name_mib[] = {CTL_KERN, KERN_PROC, KERN_PROC_ALL, 0};
|
214
|
-
|
215
|
-
rb_scan_args(argc, argv, "01", &v_pid);
|
216
|
-
|
217
|
-
// Get size of proc kproc buffer
|
218
|
-
err = sysctl( (int *) name_mib, PROC_MIB_LEN, NULL, &length, NULL, 0);
|
219
|
-
|
220
|
-
if(err == -1)
|
221
|
-
rb_raise(cProcTableError, "sysctl: %s", strerror(errno));
|
222
|
-
|
223
|
-
// Populate the kproc buffer
|
224
|
-
procs = ruby_xmalloc(length);
|
225
|
-
|
226
|
-
err = sysctl( (int *) name_mib, PROC_MIB_LEN, procs, &length, NULL, 0);
|
227
|
-
|
228
|
-
if(err == -1)
|
229
|
-
rb_raise(cProcTableError, "sysctl: %s", strerror(errno));
|
230
|
-
|
231
|
-
// If we're here, we got our list
|
232
|
-
count = length / sizeof(struct kinfo_proc);
|
233
|
-
|
234
|
-
for(i = 0; i < count; i++) {
|
235
|
-
v_tty_num = Qnil;
|
236
|
-
v_tty_dev = Qnil;
|
237
|
-
v_start_time = Qnil;
|
238
|
-
|
239
|
-
// If a PID is provided, skip unless the PID matches
|
240
|
-
if( (!NIL_P(v_pid)) && (procs[i].kp_proc.p_pid != NUM2INT(v_pid)) )
|
241
|
-
continue;
|
242
|
-
|
243
|
-
// cmdline will be set only if process exists and belongs to current user or
|
244
|
-
// current user is root
|
245
|
-
v_cmdline = Qnil;
|
246
|
-
v_exe = Qnil;
|
247
|
-
v_environ = Qnil;
|
248
|
-
argv_of_pid(procs[i].kp_proc.p_pid, &v_cmdline, &v_exe, &v_environ);
|
249
|
-
|
250
|
-
// Get the start time of the process
|
251
|
-
v_start_time = rb_time_new(
|
252
|
-
procs[i].kp_proc.p_un.__p_starttime.tv_sec,
|
253
|
-
procs[i].kp_proc.p_un.__p_starttime.tv_usec
|
254
|
-
);
|
255
|
-
|
256
|
-
// Get the state of the process
|
257
|
-
switch(procs[i].kp_proc.p_stat)
|
258
|
-
{
|
259
|
-
case SIDL:
|
260
|
-
strcpy(state, "idle");
|
261
|
-
break;
|
262
|
-
case SRUN:
|
263
|
-
strcpy(state, "run");
|
264
|
-
break;
|
265
|
-
case SSLEEP:
|
266
|
-
strcpy(state, "sleep");
|
267
|
-
break;
|
268
|
-
case SSTOP:
|
269
|
-
strcpy(state, "stop");
|
270
|
-
break;
|
271
|
-
case SZOMB:
|
272
|
-
strcpy(state, "zombie");
|
273
|
-
break;
|
274
|
-
default:
|
275
|
-
strcpy(state, "unknown");
|
276
|
-
break;
|
277
|
-
}
|
278
|
-
|
279
|
-
// Get ttynum and ttydev. If ttynum is -1, there is no tty.
|
280
|
-
if(procs[i].kp_eproc.e_tdev != -1){
|
281
|
-
v_tty_num = INT2FIX(procs[i].kp_eproc.e_tdev),
|
282
|
-
v_tty_dev = rb_str_new2(devname(procs[i].kp_eproc.e_tdev, S_IFCHR));
|
283
|
-
}
|
284
|
-
|
285
|
-
v_groups = rb_ary_new();
|
286
|
-
|
287
|
-
for (g = 0; g < procs[i].kp_eproc.e_ucred.cr_ngroups; ++g)
|
288
|
-
rb_ary_push(v_groups, INT2FIX(procs[i].kp_eproc.e_ucred.cr_groups[g]));
|
289
|
-
|
290
|
-
v_pstruct = rb_struct_new(
|
291
|
-
sProcStruct,
|
292
|
-
INT2FIX(procs[i].kp_proc.p_pid),
|
293
|
-
INT2FIX(procs[i].kp_eproc.e_ppid),
|
294
|
-
INT2FIX(procs[i].kp_eproc.e_pgid),
|
295
|
-
INT2FIX(procs[i].kp_eproc.e_pcred.p_ruid),
|
296
|
-
INT2FIX(procs[i].kp_eproc.e_pcred.p_rgid),
|
297
|
-
INT2FIX(procs[i].kp_eproc.e_ucred.cr_uid),
|
298
|
-
rb_ary_entry(v_groups, 0),
|
299
|
-
v_groups,
|
300
|
-
INT2FIX(procs[i].kp_eproc.e_pcred.p_svuid),
|
301
|
-
INT2FIX(procs[i].kp_eproc.e_pcred.p_svgid),
|
302
|
-
rb_str_new2(procs[i].kp_proc.p_comm),
|
303
|
-
rb_str_new2(state),
|
304
|
-
rb_float_new(procs[i].kp_proc.p_pctcpu),
|
305
|
-
Qnil,
|
306
|
-
v_tty_num,
|
307
|
-
v_tty_dev,
|
308
|
-
rb_str_new2(procs[i].kp_eproc.e_wmesg),
|
309
|
-
INT2FIX(procs[i].kp_proc.p_rtime.tv_sec),
|
310
|
-
INT2FIX(procs[i].kp_proc.p_priority),
|
311
|
-
INT2FIX(procs[i].kp_proc.p_usrpri),
|
312
|
-
INT2FIX(procs[i].kp_proc.p_nice),
|
313
|
-
v_cmdline,
|
314
|
-
v_exe,
|
315
|
-
v_environ,
|
316
|
-
v_start_time,
|
317
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_maxrss) : Qnil,
|
318
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_ixrss) : Qnil,
|
319
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_idrss) : Qnil,
|
320
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_isrss) : Qnil,
|
321
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_minflt) : Qnil,
|
322
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_majflt) : Qnil,
|
323
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_nswap) : Qnil,
|
324
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_inblock) : Qnil,
|
325
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_oublock) : Qnil,
|
326
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_msgsnd) : Qnil,
|
327
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_msgrcv) : Qnil,
|
328
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_nsignals) : Qnil,
|
329
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_nvcsw) : Qnil,
|
330
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_nivcsw) : Qnil,
|
331
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_utime.tv_sec) : Qnil,
|
332
|
-
(procs[i].kp_proc.p_ru && procs[i].kp_proc.p_stat != 5) ? LONG2NUM(procs[i].kp_proc.p_ru->ru_stime.tv_sec) : Qnil
|
333
|
-
);
|
334
|
-
|
335
|
-
OBJ_FREEZE(v_pstruct); // This is read-only data
|
336
|
-
|
337
|
-
if(rb_block_given_p())
|
338
|
-
rb_yield(v_pstruct);
|
339
|
-
else
|
340
|
-
rb_ary_push(v_array, v_pstruct);
|
341
|
-
}
|
342
|
-
|
343
|
-
if(procs)
|
344
|
-
free(procs);
|
345
|
-
|
346
|
-
if(!rb_block_given_p()){
|
347
|
-
if(NIL_P(v_pid))
|
348
|
-
return v_array;
|
349
|
-
else
|
350
|
-
return v_pstruct;
|
351
|
-
}
|
352
|
-
|
353
|
-
return Qnil;
|
354
|
-
}
|
355
|
-
|
356
|
-
/*
|
357
|
-
* call-seq:
|
358
|
-
* ProcTable.fields
|
359
|
-
*
|
360
|
-
* Returns an array of fields that each ProcTableStruct will contain. This
|
361
|
-
* may be useful if you want to know in advance what fields are available
|
362
|
-
* without having to perform at least one read of the /proc table.
|
363
|
-
*/
|
364
|
-
static VALUE pt_fields(VALUE klass){
|
365
|
-
VALUE v_array = rb_ary_new();
|
366
|
-
|
367
|
-
VALUE v_members = rb_struct_s_members(sProcStruct), v_member;
|
368
|
-
long size = RARRAY_LEN(v_members);
|
369
|
-
int i;
|
370
|
-
|
371
|
-
for(i = 0; i < size; i++) {
|
372
|
-
v_member = rb_funcall(rb_ary_entry(v_members, i), rb_intern("to_s"), 0);
|
373
|
-
rb_ary_push(v_array, v_member);
|
374
|
-
}
|
375
|
-
|
376
|
-
return v_array;
|
377
|
-
}
|
378
|
-
|
379
|
-
/*
|
380
|
-
* A Ruby interface for gathering process table information.
|
381
|
-
*/
|
382
|
-
void Init_proctable(){
|
383
|
-
VALUE mSys, cProcTable;
|
384
|
-
|
385
|
-
/* The Sys module serves as a namespace only */
|
386
|
-
mSys = rb_define_module("Sys");
|
387
|
-
|
388
|
-
/* The ProcTable class encapsulates process table information */
|
389
|
-
cProcTable = rb_define_class_under(mSys, "ProcTable", rb_cObject);
|
390
|
-
|
391
|
-
/* The Error class typically raised if any of the ProcTable methods fail */
|
392
|
-
cProcTableError = rb_define_class_under(cProcTable, "Error", rb_eStandardError);
|
393
|
-
|
394
|
-
/* Singleton methods */
|
395
|
-
|
396
|
-
rb_define_singleton_method(cProcTable, "ps", pt_ps, -1);
|
397
|
-
rb_define_singleton_method(cProcTable, "fields", pt_fields, 0);
|
398
|
-
|
399
|
-
/* There is no constructor */
|
400
|
-
rb_funcall(cProcTable, rb_intern("private_class_method"), 1, ID2SYM(rb_intern("new")));
|
401
|
-
|
402
|
-
/* 1.0.0: The version of the sys-proctable library */
|
403
|
-
rb_define_const(cProcTable, "VERSION", rb_str_new2("1.0.0"));
|
404
|
-
|
405
|
-
/* Structs */
|
406
|
-
|
407
|
-
sProcStruct = rb_struct_define("ProcTableStruct",
|
408
|
-
"pid", /* Process identifier */
|
409
|
-
"ppid", /* Parent process id */
|
410
|
-
"pgid", /* Process group id */
|
411
|
-
"ruid", /* Real user id */
|
412
|
-
"rgid", /* Real group id */
|
413
|
-
"euid", /* Effective user id */
|
414
|
-
"egid", /* Effective group id */
|
415
|
-
"groups", /* All effective group ids */
|
416
|
-
"svuid", /* Saved effective user id */
|
417
|
-
"svgid", /* Saved effective group id */
|
418
|
-
"comm", /* Command name (15 chars) */
|
419
|
-
"state", /* Process status */
|
420
|
-
"pctcpu", /* %cpu for this process during p_swtime */
|
421
|
-
"oncpu", /* nil */
|
422
|
-
"tnum", /* Controlling tty dev */
|
423
|
-
"tdev", /* Controlling tty name */
|
424
|
-
"wmesg", /* Wchan message */
|
425
|
-
"rtime", /* Real time */
|
426
|
-
"priority", /* Process priority */
|
427
|
-
"usrpri", /* User-priority */
|
428
|
-
"nice", /* Process "nice" value */
|
429
|
-
"cmdline", /* Complete command line */
|
430
|
-
"exe", /* Saved pathname of the executed command */
|
431
|
-
"environ", /* Hash with process environment variables */
|
432
|
-
"starttime", /* Process start time */
|
433
|
-
"maxrss", /* Max resident set size (PL) */
|
434
|
-
"ixrss", /* Integral shared memory size (NU) */
|
435
|
-
"idrss", /* Integral unshared data (NU) */
|
436
|
-
"isrss", /* Integral unshared stack (NU) */
|
437
|
-
"minflt", /* Page reclaims (NU) */
|
438
|
-
"majflt", /* Page faults (NU) */
|
439
|
-
"nswap", /* Swaps (NU) */
|
440
|
-
"inblock", /* Block input operations (atomic) */
|
441
|
-
"oublock", /* Block output operations (atomic) */
|
442
|
-
"msgsnd", /* Messages sent (atomic) */
|
443
|
-
"msgrcv", /* Messages received (atomic) */
|
444
|
-
"nsignals", /* Signals received (atomic) */
|
445
|
-
"nvcsw", /* Voluntary context switches (atomic) */
|
446
|
-
"nivcsw", /* Involuntary context switches (atomic) */
|
447
|
-
"utime", /* User time used (PL) */
|
448
|
-
"stime", /* System time used (PL) */
|
449
|
-
NULL
|
450
|
-
);
|
451
|
-
}
|