| 1 |
#include <stdio.h> |
| 2 |
#include <stdlib.h> |
| 3 |
#include <unistd.h> |
| 4 |
#include <strings.h> |
| 5 |
#include <pwd.h> |
| 6 |
#include "statgrab.h" |
| 7 |
|
| 8 |
char usage(char *progname){ |
| 9 |
fprintf(stderr," Usage: %s [-efAnh] [-u user] [-o sort]\n\n", progname); |
| 10 |
fprintf(stderr, "\t-h\tDisplays this help\n"); |
| 11 |
fprintf(stderr, "\t-e\tAll processes\n"); |
| 12 |
fprintf(stderr, "\t-f\tFull listing\n"); |
| 13 |
fprintf(stderr, "\t-a\tAll processes (same as -e)\n"); |
| 14 |
fprintf(stderr, "\t-n\tDont look up usernames\n"); |
| 15 |
fprintf(stderr, "\t-u\tLook for just this user\n"); |
| 16 |
fprintf(stderr, "\t-o\tSorting method. Valid are cpu, mem, pid, uid, gid and size\n"); |
| 17 |
fprintf(stderr, "\n"); |
| 18 |
exit(1); |
| 19 |
} |
| 20 |
|
| 21 |
char *size_conv(long long number){ |
| 22 |
char type[] = {'B', 'K', 'M', 'G', 'T'}; |
| 23 |
int x=0; |
| 24 |
static char string[10]; |
| 25 |
|
| 26 |
for(;x<5;x++){ |
| 27 |
if( (number/1024) < (100)) { |
| 28 |
break; |
| 29 |
} |
| 30 |
number = (number/1024); |
| 31 |
} |
| 32 |
|
| 33 |
snprintf(string, 10, "%lld%c", number, type[x]); |
| 34 |
return string; |
| 35 |
|
| 36 |
} |
| 37 |
|
| 38 |
char *username(uid_t uid){ |
| 39 |
static struct passwd *pass; |
| 40 |
|
| 41 |
pass = getpwuid(uid); |
| 42 |
if(pass == NULL){ |
| 43 |
return "NULL"; |
| 44 |
} |
| 45 |
|
| 46 |
return pass->pw_name; |
| 47 |
} |
| 48 |
|
| 49 |
char *proc_state(sg_process_state state){ |
| 50 |
static char *pstat = "Unk"; |
| 51 |
|
| 52 |
if(state == SG_PROCESS_STATE_RUNNING){ |
| 53 |
pstat="Run"; |
| 54 |
} |
| 55 |
|
| 56 |
if(state == SG_PROCESS_STATE_SLEEPING){ |
| 57 |
pstat="Sleep"; |
| 58 |
} |
| 59 |
|
| 60 |
if(state == SG_PROCESS_STATE_STOPPED){ |
| 61 |
pstat="Stop"; |
| 62 |
} |
| 63 |
|
| 64 |
if(state == SG_PROCESS_STATE_ZOMBIE){ |
| 65 |
pstat="Zomb"; |
| 66 |
} |
| 67 |
|
| 68 |
if(state == SG_PROCESS_STATE_UNKNOWN){ |
| 69 |
pstat="Unk"; |
| 70 |
} |
| 71 |
|
| 72 |
return pstat; |
| 73 |
} |
| 74 |
|
| 75 |
int main(int argc, char **argv){ |
| 76 |
extern char *optarg; |
| 77 |
int c; |
| 78 |
int x; |
| 79 |
int full=0; |
| 80 |
int detailed=0; |
| 81 |
int nolookup=0; |
| 82 |
struct passwd *pwd_ent; |
| 83 |
uid_t uid = -1; |
| 84 |
|
| 85 |
int (*sortby_ptr)(const void *va, const void *vb); |
| 86 |
|
| 87 |
int entries; |
| 88 |
extern char *optarg; |
| 89 |
|
| 90 |
|
| 91 |
sg_process_stats *sg_proc = NULL; |
| 92 |
/* Default behaviour - sort by pid */ |
| 93 |
sortby_ptr = sg_process_compare_pid; |
| 94 |
|
| 95 |
while ((c = getopt(argc, argv, "hneAfU:u:o:")) != -1){ |
| 96 |
switch (c){ |
| 97 |
case 'e': |
| 98 |
case 'A': |
| 99 |
full=1; |
| 100 |
break; |
| 101 |
case 'f': |
| 102 |
detailed=1; |
| 103 |
break; |
| 104 |
case 'n': |
| 105 |
nolookup=1; |
| 106 |
break; |
| 107 |
case 'U': |
| 108 |
case 'u': |
| 109 |
pwd_ent = getpwnam(optarg); |
| 110 |
if (pwd_ent == NULL){ |
| 111 |
fprintf(stderr, "Error: user %s does not exist\n", optarg); |
| 112 |
exit(1); |
| 113 |
} |
| 114 |
uid = pwd_ent->pw_uid; |
| 115 |
break; |
| 116 |
case 'o': |
| 117 |
if(!strncasecmp(optarg, "cpu", 3)){ |
| 118 |
sortby_ptr = sg_process_compare_cpu; |
| 119 |
break; |
| 120 |
} |
| 121 |
if(!strncasecmp(optarg, "size", 3)){ |
| 122 |
sortby_ptr = sg_process_compare_size; |
| 123 |
break; |
| 124 |
} |
| 125 |
if(!strncasecmp(optarg, "pid", 3)){ |
| 126 |
sortby_ptr = sg_process_compare_pid; |
| 127 |
break; |
| 128 |
} |
| 129 |
if(!strncasecmp(optarg, "mem", 3)){ |
| 130 |
sortby_ptr = sg_process_compare_res; |
| 131 |
break; |
| 132 |
} |
| 133 |
if(!strncasecmp(optarg, "res", 3)){ |
| 134 |
sortby_ptr = sg_process_compare_res; |
| 135 |
break; |
| 136 |
} |
| 137 |
if(!strncasecmp(optarg, "uid", 3)){ |
| 138 |
sortby_ptr = sg_process_compare_uid; |
| 139 |
break; |
| 140 |
} |
| 141 |
if(!strncasecmp(optarg, "gid", 3)){ |
| 142 |
sortby_ptr = sg_process_compare_gid; |
| 143 |
break; |
| 144 |
} |
| 145 |
case 'h': |
| 146 |
default: |
| 147 |
usage(argv[0]); |
| 148 |
break; |
| 149 |
exit(1); |
| 150 |
} |
| 151 |
} |
| 152 |
|
| 153 |
/* Get the procs - and sort them */ |
| 154 |
sg_proc = sg_get_process_stats(&entries); |
| 155 |
qsort(sg_proc, entries, sizeof *sg_proc, sortby_ptr); |
| 156 |
|
| 157 |
/* Print the headers out on stderr, so should someone want to use this in a script, they wont need to |
| 158 |
* strip out the first line. |
| 159 |
*/ |
| 160 |
if (full){ |
| 161 |
fprintf(stderr, "%9s %6s %6s %6s %7s %7s %4s %s\n", "User", "pid", "parent", "state", "size", "res", "CPU", "Process"); |
| 162 |
}else{ |
| 163 |
fprintf(stderr, "%6s %7s %4s %s\n", "pid", "size", "CPU", "Process"); |
| 164 |
} |
| 165 |
|
| 166 |
for(x=0; x<entries; x++){ |
| 167 |
char *proc_out=NULL; |
| 168 |
if (detailed) { |
| 169 |
proc_out = sg_proc->proctitle; |
| 170 |
}else{ |
| 171 |
proc_out = sg_proc->process_name; |
| 172 |
} |
| 173 |
|
| 174 |
if((uid == -1) || (uid == sg_proc->uid)){ |
| 175 |
if(full){ |
| 176 |
printf("%9s %6d %6d %6s %7s %7s %2.2f %s\n", username(sg_proc->uid), (int)sg_proc->pid, (int)sg_proc->parent, proc_state(sg_proc->state), size_conv(sg_proc->proc_size), size_conv(sg_proc->proc_resident), sg_proc->cpu_percent, proc_out); |
| 177 |
}else{ |
| 178 |
printf("%6d %7s %2.2f %s\n", (int)sg_proc->pid, size_conv(sg_proc->proc_size), sg_proc->cpu_percent, proc_out); |
| 179 |
} |
| 180 |
} |
| 181 |
|
| 182 |
sg_proc++; |
| 183 |
} |
| 184 |
|
| 185 |
return 0; |
| 186 |
} |