1
0
mirror of https://github.com/Llewellynvdm/conky.git synced 2025-01-11 18:38:45 +00:00

new top routines using linked lists instead of arrays

git-svn-id: https://conky.svn.sourceforge.net/svnroot/conky/trunk/conky@384 7f574dfc-610e-0410-a909-a81674777703
This commit is contained in:
David Carter 2005-11-10 01:20:19 +00:00
parent 39dd056722
commit 9b2401b36e
3 changed files with 136 additions and 114 deletions

View File

@ -205,8 +205,6 @@ void update_stuff()
if (NEED(INFO_MAIL)) if (NEED(INFO_MAIL))
update_mail_count(); update_mail_count();
if (NEED(INFO_TOP))
update_top();
#if defined(__linux__) #if defined(__linux__)
if (NEED(INFO_I8K)) if (NEED(INFO_I8K))
@ -231,6 +229,7 @@ void update_stuff()
if (NEED(INFO_LOADAVG)) if (NEED(INFO_LOADAVG))
update_load_average(); update_load_average();
if ((NEED(INFO_MEM) || NEED(INFO_BUFFERS)) && if ((NEED(INFO_MEM) || NEED(INFO_BUFFERS)) &&
current_update_time - last_meminfo_update > 6.9) { current_update_time - last_meminfo_update > 6.9) {
update_meminfo(); update_meminfo();
@ -239,6 +238,9 @@ void update_stuff()
last_meminfo_update = current_update_time; last_meminfo_update = current_update_time;
} }
if (NEED(INFO_TOP))
update_top();
/* update_fs_stat() won't do anything if there aren't fs -things */ /* update_fs_stat() won't do anything if there aren't fs -things */
if (NEED(INFO_FS) && current_update_time - last_fs_update > 12.9) { if (NEED(INFO_FS) && current_update_time - last_fs_update > 12.9) {
update_fs_stats(); update_fs_stats();

231
src/top.c
View File

@ -13,6 +13,8 @@ static unsigned int g_time = 0;
static unsigned long previous_total = 0; static unsigned long previous_total = 0;
static struct process *first_process = 0; static struct process *first_process = 0;
static struct process *find_process(pid_t pid) static struct process *find_process(pid_t pid)
{ {
struct process *p = first_process; struct process *p = first_process;
@ -169,8 +171,9 @@ static int process_parse_stat(struct process *process)
if (!cur->memmax) if (!cur->memmax)
update_total_processes(); update_total_processes();
process->totalmem = ((float) process->rss / cur->memmax) / 10;
process->totalmem = (float)(((float) process->rss / cur->memmax) / 10);
if (process->previous_user_time == ULONG_MAX) if (process->previous_user_time == ULONG_MAX)
process->previous_user_time = process->user_time; process->previous_user_time = process->user_time;
if (process->previous_kernel_time == ULONG_MAX) if (process->previous_kernel_time == ULONG_MAX)
@ -375,122 +378,138 @@ inline static void calc_cpu_each(unsigned long total)
/* Find the top processes */ /* Find the top processes */
/******************************************/ /******************************************/
/* /*
* Result is stored in decreasing order in best[0-9]. * cpu comparison function for insert_sp_element
*/
int compare_cpu(struct process *a, struct process *b) {
if (a->amount < b->amount) return 1;
return 0;
}
/*
* mem comparison function for insert_sp_element
*/
int compare_mem(struct process *a, struct process *b) {
if (a->totalmem < b->totalmem) return 1;
return 0;
}
/*
* insert this process into the list in a sorted fashion,
* or destroy it if it doesn't fit on the list
*/
int insert_sp_element(
struct sorted_process * sp_cur
, struct sorted_process ** p_sp_head
, struct sorted_process ** p_sp_tail
, int max_elements
, int (*compare_funct) (struct process *, struct process *)
) {
struct sorted_process * sp_readthru=NULL, * sp_destroy=NULL;
int did_insert = 0, x = 0;
if (*p_sp_head == NULL) {
*p_sp_head = sp_cur;
*p_sp_tail = sp_cur;
return(1);
}
for(sp_readthru=*p_sp_head, x=0; sp_readthru != NULL && x < max_elements; sp_readthru=sp_readthru->less, x++) {
if (compare_funct(sp_readthru->proc, sp_cur->proc) && !did_insert) {
/* sp_cur is bigger than sp_readthru so insert it before sp_readthru */
sp_cur->less=sp_readthru;
if (sp_readthru == *p_sp_head) {
*p_sp_head = sp_cur; /* insert as the new head of the list */
} else {
sp_readthru->greater->less = sp_cur; /* insert inside the list */
sp_cur->greater = sp_readthru->greater;
}
sp_readthru->greater=sp_cur;
did_insert = ++x; /* element was inserted, so increase the counter */
}
}
if (x < max_elements && sp_readthru == NULL && !did_insert) {
/* sp_cur is the smallest element and list isn't full, so insert at the end */
(*p_sp_tail)->less=sp_cur;
sp_cur->greater=*p_sp_tail;
*p_sp_tail = sp_cur;
did_insert=x;
} else if (x==max_elements && sp_readthru != NULL) {
/* we inserted an element and now the list is too big by one. Destroy the smallest element */
sp_destroy = sp_readthru;
sp_readthru->greater->less = NULL;
*p_sp_tail = sp_readthru->greater;
free(sp_destroy);
}
if (!did_insert) {
/* sp_cur wasn't added to the sorted list, so destroy it */
free(sp_cur);
}
return did_insert;
}
/*
* create a new sp_process structure
*/ */
#define MAX_TOP_SIZE 400 /* this is plenty big */ struct sorted_process * malloc_sp(struct process * proc) {
static struct process **sorttmp; struct sorted_process * sp;
static size_t sorttmp_size = 1; sp = malloc(sizeof(struct sorted_process));
sp->greater = NULL;
sp->less = NULL;
sp->proc = proc;
return(sp);
}
/*
* copy the procs in the sorted list to the array, and destroy the list
*/
void sp_acopy(struct sorted_process *sp_head, struct process ** ar, int max_size) {
int comparecpu(const void * a, const void * b) struct sorted_process * sp_cur, * sp_tmp;
{ int x;
if ((*(struct process **)a)->amount > (*(struct process **)b)->amount) {
return -1; sp_cur = sp_head;
for (x=0; x < max_size && sp_cur != NULL; x++) {
ar[x] = sp_cur->proc;
sp_tmp = sp_cur;
sp_cur= sp_cur->less;
free(sp_tmp);
} }
if ((*(struct process **)a)->amount < (*(struct process **)b)->amount) {
return 1;
}
return 0;
} }
int comparemem(const void * a, const void * b) /* ****************************************************************** */
{ /* Get a sorted list of the top cpu hogs and top mem hogs. */
if ((*(struct process **)a)->totalmem > (*(struct process **)b)->totalmem) { /* Results are stored in the cpu,mem arrays in decreasing order[0-9]. */
return -1; /* ****************************************************************** */
}
if ((*(struct process **)a)->totalmem < (*(struct process **)b)->totalmem) {
return 1;
}
return 0;
}
inline void process_find_top(struct process **cpu, struct process **mem) inline void process_find_top(struct process **cpu, struct process **mem)
{ {
struct process *pr; struct sorted_process *spc_head=NULL, *spc_tail=NULL, *spc_cur=NULL;
if (sorttmp == NULL) { struct sorted_process *spm_head=NULL, *spm_tail=NULL, *spm_cur=NULL;
sorttmp = malloc(sizeof(struct process) * sorttmp_size); struct process *cur_proc=NULL;
assert(sorttmp != NULL); unsigned long total =0;
}
unsigned long total = 0; if (!top_cpu && !top_mem) return;
unsigned long i, j;
total = calc_cpu_total(); /* calculate the total of the processor */ total = calc_cpu_total(); /* calculate the total of the processor */
update_process_table(); /* update the table with process list */
calc_cpu_each(total); /* and then the percentage for each task */
process_cleanup(); /* cleanup list from exited processes */
update_meminfo();
cur_proc = first_process;
update_process_table(); /* update the table with process list */ while (cur_proc !=NULL) {
calc_cpu_each(total); /* and then the percentage for each task */ if (top_cpu) {
process_cleanup(); /* cleanup list from exited processes */ spc_cur = malloc_sp(cur_proc);
insert_sp_element(spc_cur, &spc_head, &spc_tail, MAX_SP, &compare_cpu);
/* }
* this is really ugly, if (top_mem) {
* not to mention probably not too efficient. spm_cur = malloc_sp(cur_proc);
* the main problem is that there could be any number of processes, insert_sp_element(spm_cur, &spm_head, &spm_tail, MAX_SP, &compare_mem);
* however we have to use a fixed size for the "best" array.
* right now i can't think of a better way to do this,
* although i'm sure there is one.
* Perhaps just using a linked list would be more effecient?
* I'm too fucking lazy to do that right now.
*/
if (top_cpu) {
pr = first_process;
i = 0;
while (pr) {
if (i < sorttmp_size && pr->counted) {
sorttmp[i] = pr;
i++;
} else if (i == sorttmp_size && pr->counted && sorttmp_size < MAX_TOP_SIZE) {
sorttmp_size++;
sorttmp =
realloc(sorttmp,
sizeof(struct process) *
sorttmp_size);
sorttmp[i] = pr;
i++;
}
pr = pr->next;
}
if (i + 1 < sorttmp_size) {
sorttmp_size--;
sorttmp =
realloc(sorttmp,
sizeof(struct process) * sorttmp_size);
}
qsort(sorttmp, i, sizeof(struct process *), comparecpu);
for (i = 0; i < 10; i++) {
cpu[i] = sorttmp[i];
} }
cur_proc = cur_proc->next;
} }
if (top_mem) { sp_acopy(spc_head, cpu, MAX_SP);
pr = first_process; sp_acopy(spm_head, mem, MAX_SP);
i = 0; }
while (pr) {
if (i < sorttmp_size && pr->counted) {
sorttmp[i] = pr;
i++;
} else if (i == sorttmp_size && pr->counted && sorttmp_size < MAX_TOP_SIZE) {
sorttmp_size++;
sorttmp =
realloc(sorttmp,
sizeof(struct process) *
sorttmp_size);
sorttmp[i] = pr;
i++;
}
pr = pr->next;
}
if (i + 1 < sorttmp_size) {
sorttmp_size--;
sorttmp =
realloc(sorttmp,
sizeof(struct process) * sorttmp_size);
}
qsort(sorttmp, i, sizeof(struct process *), comparemem);
sorttmp_size = i;
for (i = 0, j = 0; i < sorttmp_size && j < 10; i++) {
if (j == 0 || sorttmp[i]->totalmem != mem[j-1]->totalmem
|| strncmp(sorttmp[i]->name, mem[j-1]->name,128)) {
mem[j++] = sorttmp[i];
}
}
}
}

View File

@ -64,22 +64,23 @@
#define PROCFS_TEMPLATE "/proc/%d/stat" #define PROCFS_TEMPLATE "/proc/%d/stat"
#define PROCFS_TEMPLATE_MEM "/proc/%d/statm" #define PROCFS_TEMPLATE_MEM "/proc/%d/statm"
#define PROCFS_CMDLINE_TEMPLATE "/proc/%d/cmdline" #define PROCFS_CMDLINE_TEMPLATE "/proc/%d/cmdline"
#define MAX_SP 10 //number of elements to sort
/******************************************/ /******************************************/
/* Globals */ /* Globals */
/******************************************/ /******************************************/
/******************************************/ /******************************************/
/* Process class */ /* Process class */
/******************************************/ /******************************************/
struct sorted_process {
struct sorted_process *greater;
struct sorted_process *less;
struct process *proc;
};
/* /*
* Pointer to head of process list * Pointer to head of process list
*/ */