evisum/src/bin/ui/ui_process_view.c

1489 lines
42 KiB
C

#include "ui_process_view.h"
#include "../system/process.h"
#include "util.c"
typedef struct
{
Evas_Object *win;
Evas_Object *content;
Evas_Object *tab_main;
Evas_Object *tab_children;
Evas_Object *tab_thread;
Evas_Object *tab_manual;
Evas_Object *main_view;
Evas_Object *children_view;
Evas_Object *thread_view;
Evas_Object *manual_view;
Evas_Object *current_view;
Evas_Object *entry_pid_cmd;
Evas_Object *entry_pid_cmd_args;
Evas_Object *entry_pid_user;
Evas_Object *entry_pid_pid;
Evas_Object *entry_pid_ppid;
Evas_Object *entry_pid_uid;
Evas_Object *entry_pid_cpu;
Evas_Object *entry_pid_threads;
Evas_Object *entry_pid_virt;
Evas_Object *entry_pid_rss;
Evas_Object *entry_pid_shared;
Evas_Object *entry_pid_size;
Evas_Object *entry_pid_started;
Evas_Object *entry_pid_nice;
Evas_Object *entry_pid_pri;
Evas_Object *entry_pid_state;
Evas_Object *entry_pid_cpu_usage;
Evas_Object *btn_start;
Evas_Object *btn_stop;
Evas_Object *btn_kill;
int64_t pid_cpu_time;
Evas_Object *genlist_children;
Evas_Object *genlist_threads;
Eina_Hash *hash_cpu_times;
Evisum_Ui_Cache *cache;
Evas_Object *tab_thread_id;
Evas_Object *tab_thread_name;
Evas_Object *tab_thread_state;
Evas_Object *tab_thread_cpu_id;
Evas_Object *tab_thread_cpu_usage;
Eina_Bool sort_reverse;
Evas_Object *entry_manual;
Eina_Bool manual_init;
int poll_delay;
int64_t start;
char *selected_cmd;
int selected_pid;
int (*sort_cb)(const void *p1, const void *p2);
Ecore_Thread *thread;
} Ui_Data;
typedef struct
{
int tid;
char *name;
char *state;
int cpu_id;
double cpu_usage;
} Thread_Info;
static Thread_Info *
_thread_info_new(Proc_Info *thr, double cpu_usage)
{
Thread_Info *t = calloc(1, sizeof(Thread_Info));
if (!t) return NULL;
t->tid = thr->tid;
t->name = strdup(thr->thread_name);
t->state = strdup(thr->state);
t->cpu_id = thr->cpu_id;
t->cpu_usage = cpu_usage;
return t;
}
Eina_List *
_exe_response(const char *command)
{
FILE *p;
Eina_List *lines;
char buf[1024];
p = popen(command, "r");
if (!p)
return NULL;
lines = NULL;
while ((fgets(buf, sizeof(buf), p)) != NULL)
{
lines = eina_list_append(lines, _man2entry(buf));
}
pclose(p);
return lines;
}
static void
_item_unrealized_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd;
Evas_Object *o;
Eina_List *contents = NULL;
pd = data;
elm_genlist_item_all_contents_unset(event_info, &contents);
EINA_LIST_FREE(contents, o)
{
evisum_ui_item_cache_item_release(pd->cache, o);
}
}
static void
_item_del(void *data, Evas_Object *obj EINA_UNUSED)
{
Thread_Info *t = data;
if (t->name)
free(t->name);
if (t->state)
free(t->state);
free(t);
}
static Evas_Object *
_item_column_add(Evas_Object *tbl, const char *text, int col)
{
Evas_Object *rec, *lb;
lb = elm_label_add(tbl);
evas_object_size_hint_weight_set(lb, EXPAND, EXPAND);
evas_object_size_hint_align_set(lb, 0.0, FILL);
evas_object_data_set(tbl, text, lb);
evas_object_show(lb);
rec = evas_object_rectangle_add(tbl);
evas_object_data_set(lb, "rect", rec);
elm_table_pack(tbl, lb, col, 0, 1, 1);
elm_table_pack(tbl, rec, col, 0, 1, 1);
return lb;
}
static Evas_Object *
_item_create(Evas_Object *parent)
{
Evas_Object *tbl, *rec, *lb, *pb;
tbl = elm_table_add(parent);
evas_object_size_hint_weight_set(tbl, EXPAND, EXPAND);
evas_object_size_hint_align_set(tbl, FILL, FILL);
evas_object_show(tbl);
lb = _item_column_add(tbl, "tid", 0);
evas_object_size_hint_align_set(lb, 0.5, FILL);
_item_column_add(tbl, "name", 1);
lb = _item_column_add(tbl, "state", 2);
evas_object_size_hint_align_set(lb, 0.5, FILL);
lb = _item_column_add(tbl, "cpu_id", 3);
evas_object_size_hint_align_set(lb, 0.5, FILL);
pb = elm_progressbar_add(parent);
evas_object_size_hint_align_set(pb, FILL, FILL);
evas_object_size_hint_weight_set(pb, EXPAND, EXPAND);
elm_progressbar_unit_format_set(pb, "%1.1f %%");
evas_object_data_set(tbl, "cpu_usage", pb);
rec = evas_object_rectangle_add(tbl);
evas_object_data_set(pb, "rect", rec);
elm_table_pack(tbl, pb, 4, 0, 1, 1);
elm_table_pack(tbl, rec, 4, 0, 1, 1);
return tbl;
}
static Evas_Object *
_content_get(void *data, Evas_Object *obj, const char *source)
{
Ui_Data *pd;
Thread_Info *th;
Evas_Object *lb, *rec, *pb;
Evas_Coord w, ow;
th = (void *) data;
if (strcmp(source, "elm.swallow.content")) return NULL;
if (!th) return NULL;
pd = evas_object_data_get(obj, "ui");
if (!pd) return NULL;
Item_Cache *it = evisum_ui_item_cache_item_get(pd->cache);
if (!it)
{
fprintf(stderr, "Error: Object cache creation failed.\n");
exit(-1);
}
evas_object_geometry_get(pd->tab_thread_id, NULL, NULL, &w, NULL);
lb = evas_object_data_get(it->obj, "tid");
elm_object_text_set(lb, eina_slstr_printf("%d", th->tid));
evas_object_geometry_get(lb, NULL, NULL, &ow, NULL);
if (ow > w) evas_object_size_hint_min_set(pd->tab_thread_id, w, 1);
rec = evas_object_data_get(lb, "rect");
evas_object_size_hint_min_set(rec, w, 1);
evas_object_geometry_get(pd->tab_thread_name, NULL, NULL, &w, NULL);
lb = evas_object_data_get(it->obj, "name");
elm_object_text_set(lb, eina_slstr_printf("%s", th->name));
evas_object_geometry_get(lb, NULL, NULL, &ow, NULL);
if (ow > w) evas_object_size_hint_min_set(pd->tab_thread_name, w, 1);
rec = evas_object_data_get(lb, "rect");
evas_object_size_hint_min_set(rec, w, 1);
evas_object_geometry_get(pd->tab_thread_state, NULL, NULL, &w, NULL);
lb = evas_object_data_get(it->obj, "state");
elm_object_text_set(lb, eina_slstr_printf("%s", th->state));
evas_object_geometry_get(lb, NULL, NULL, &ow, NULL);
if (ow > w) evas_object_size_hint_min_set(pd->tab_thread_state, w, 1);
rec = evas_object_data_get(lb, "rect");
evas_object_size_hint_min_set(rec, w, 1);
evas_object_geometry_get(pd->tab_thread_cpu_id, NULL, NULL, &w, NULL);
lb = evas_object_data_get(it->obj, "cpu_id");
elm_object_text_set(lb, eina_slstr_printf("%d", th->cpu_id));
evas_object_geometry_get(lb, NULL, NULL, &ow, NULL);
if (ow > w) evas_object_size_hint_min_set(pd->tab_thread_cpu_id, w, 1);
rec = evas_object_data_get(lb, "rect");
evas_object_size_hint_min_set(rec, w, 1);
evas_object_geometry_get(pd->tab_thread_cpu_usage, NULL, NULL, &w, NULL);
pb = evas_object_data_get(it->obj, "cpu_usage");
elm_progressbar_value_set(pb, th->cpu_usage / 100.0);
evas_object_show(pb);
rec = evas_object_data_get(pb, "rect");
evas_object_size_hint_min_set(rec, w, 1);
return it->obj;
}
static void
_genlist_ensure_n_items(Evas_Object *genlist, unsigned int items)
{
Elm_Object_Item *it;
Elm_Genlist_Item_Class *itc;
unsigned int i, existing = elm_genlist_items_count(genlist);
if (items < existing)
{
for (i = existing - items; i > 0; i--)
{
it = elm_genlist_last_item_get(genlist);
if (it)
elm_object_item_del(it);
}
}
if (items == existing) return;
itc = elm_genlist_item_class_new();
itc->item_style = "full";
itc->func.text_get = NULL;
itc->func.content_get = _content_get;
itc->func.filter_get = NULL;
itc->func.del = _item_del;
for (i = existing; i < items; i++)
{
elm_genlist_item_append(genlist, itc, NULL, NULL,
ELM_GENLIST_ITEM_NONE, NULL, NULL);
}
elm_genlist_item_class_free(itc);
}
static int
_sort_by_cpu_usage(const void *p1, const void *p2)
{
const Thread_Info *inf1, *inf2;
double one, two;
inf1 = p1; inf2 = p2;
one = inf1->cpu_usage; two = inf2->cpu_usage;
if (one > two)
return 1;
else if (one < two)
return -1;
else return 0;
}
static int
_sort_by_cpu_id(const void *p1, const void *p2)
{
const Thread_Info *inf1, *inf2;
inf1 = p1; inf2 = p2;
return inf1->cpu_id - inf2->cpu_id;
}
static int
_sort_by_state(const void *p1, const void *p2)
{
const Thread_Info *inf1, *inf2;
inf1 = p1; inf2 = p2;
return strcmp(inf1->state, inf2->state);
}
static int
_sort_by_name(const void *p1, const void *p2)
{
const Thread_Info *inf1, *inf2;
inf1 = p1; inf2 = p2;
return strcmp(inf1->name, inf2->name);
}
static int
_sort_by_tid(const void *p1, const void *p2)
{
const Thread_Info *inf1, *inf2;
inf1 = p1; inf2 = p2;
return inf1->tid - inf2->tid;
}
static void
_hash_free_cb(void *data)
{
long *cpu_time = data;
if (cpu_time)
free(cpu_time);
}
static void
_thread_info_set(Ui_Data *pd, Proc_Info *proc)
{
Proc_Info *p;
Thread_Info *t;
Elm_Object_Item *it;
Eina_List *l, *threads = NULL;
if (!pd->hash_cpu_times)
pd->hash_cpu_times = eina_hash_string_superfast_new(_hash_free_cb);
_genlist_ensure_n_items(pd->genlist_threads, eina_list_count(proc->threads));
EINA_LIST_FOREACH(proc->threads, l, p)
{
long *cpu_time, *cpu_time_prev;
double cpu_usage = 0.0;
const char *key = eina_slstr_printf("%s:%d", p->thread_name, p->tid);
if ((cpu_time_prev = eina_hash_find(pd->hash_cpu_times, key)) == NULL)
{
cpu_time = malloc(sizeof(long));
*cpu_time = p->cpu_time;
eina_hash_add(pd->hash_cpu_times, key, cpu_time);
}
else
{
cpu_usage = (double) (p->cpu_time - *cpu_time_prev)
/ pd->poll_delay;
*cpu_time_prev = p->cpu_time;
}
t = _thread_info_new(p, cpu_usage);
if (t)
threads = eina_list_append(threads, t);
}
if (pd->sort_cb)
threads = eina_list_sort(threads, eina_list_count(threads), pd->sort_cb);
if (pd->sort_reverse)
threads = eina_list_reverse(threads);
it = elm_genlist_first_item_get(pd->genlist_threads);
EINA_LIST_FREE(threads, t)
{
if (!it)
_item_del(t, NULL);
else
{
Thread_Info *prev = elm_object_item_data_get(it);
if (prev)
_item_del(prev, NULL);
elm_object_item_data_set(it, t);
elm_genlist_item_update(it);
it = elm_genlist_item_next_get(it);
}
}
}
static char *
_time_string(int64_t epoch)
{
struct tm *info;
time_t rawtime;
char buf[256];
rawtime = (time_t) epoch;
info = localtime(&rawtime);
strftime(buf, sizeof(buf), "%F %T", info);
return strdup(buf);
}
static void
_item_children_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info)
{
Elm_Object_Item *it;
Proc_Info *proc;
it = event_info;
elm_genlist_item_selected_set(it, EINA_FALSE);
proc = elm_object_item_data_get(it);
if (!proc) return;
ui_process_view_win_add(proc->pid, PROC_VIEW_DEFAULT);
}
static char *
_children_text_get(void *data, Evas_Object *obj, const char *part)
{
Proc_Info *child = data;
char buf[256];
snprintf(buf, sizeof(buf), "%s (%d) ", child->command, child->pid);
return strdup(buf);
}
static Evas_Object *
_children_icon_get(void *data, Evas_Object *obj, const char *part)
{
Proc_Info *proc;
Evas_Object *ic = elm_icon_add(obj);
proc = data;
if (!strcmp(part, "elm.swallow.icon"))
{
elm_icon_standard_set(ic,
evisum_icon_path_get(
evisum_icon_cache_find(proc)));
}
evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1);
return ic;
}
static void
_children_del(void *data, Evas_Object *obj EINA_UNUSED)
{
Proc_Info *proc = data;
eina_list_free(proc->children);
proc_info_free(proc);
}
static int
_sort_by_age(const void *p1, const void *p2)
{
const Proc_Info *c1 = p1, *c2 = p2;
return c1->start - c2->start;
}
static void
_children_populate(Evas_Object *genlist_children, Elm_Object_Item *parent,
Eina_List *children)
{
Elm_Genlist_Item_Class *itc;
Eina_List *l;
Elm_Object_Item *it;
Proc_Info *child;
itc = elm_genlist_item_class_new();
itc->item_style = "default";
itc->func.content_get = _children_icon_get;
itc->func.text_get = _children_text_get;
itc->func.filter_get = NULL;
itc->func.del = _children_del;
EINA_LIST_FOREACH(children, l, child)
{
it = elm_genlist_item_append(genlist_children, itc, child, parent,
(child->children ?
ELM_GENLIST_ITEM_TREE :
ELM_GENLIST_ITEM_NONE), NULL, NULL);
elm_genlist_item_update(it);
if (child->children)
{
child->children = eina_list_sort(child->children,
eina_list_count(child->children),
_sort_by_age);
_children_populate(genlist_children, it, child->children);
}
}
elm_genlist_item_class_free(itc);
}
static Eina_Bool
_children_view_update(void *data)
{
Eina_List *children, *l;
Proc_Info *child;
Ui_Data *pd = data;
elm_genlist_clear(pd->genlist_children);
if (pd->selected_pid == 0) return EINA_FALSE;
children = proc_info_pid_children_get(pd->selected_pid);
EINA_LIST_FOREACH(children, l, child)
{
if (child->pid == pd->selected_pid)
{
child->children = eina_list_sort(child->children,
eina_list_count(child->children),
_sort_by_age);
_children_populate(pd->genlist_children, NULL, child->children);
break;
}
}
child = eina_list_nth(children, 0);
if (child)
proc_info_free(child);
return EINA_TRUE;
}
static void
_proc_info_main(void *data, Ecore_Thread *thread)
{
Ui_Data *pd = data;
while (!ecore_thread_check(thread))
{
Proc_Info *proc = proc_info_by_pid(pd->selected_pid);
ecore_thread_feedback(thread, proc);
for (int i = 0; i < (8 * pd->poll_delay); i++)
{
if (ecore_thread_check(thread))
return;
usleep(125000);
}
}
}
static void
_proc_gone(Ui_Data *pd)
{
const char *fmt = _("%s (%d) - Not running");
elm_win_title_set(pd->win,
eina_slstr_printf(fmt,
pd->selected_cmd,
pd->selected_pid));
elm_object_disabled_set(pd->btn_start, EINA_TRUE);
elm_object_disabled_set(pd->btn_stop, EINA_TRUE);
elm_object_disabled_set(pd->btn_kill, EINA_TRUE);
if (!ecore_thread_check(pd->thread))
ecore_thread_cancel(pd->thread);
pd->thread = NULL;
}
static void
_proc_info_feedback_cb(void *data, Ecore_Thread *thread, void *msg)
{
Ui_Data *pd;
struct passwd *pwd_entry;
Proc_Info *proc;
double cpu_usage = 0.0;
pd = data;
proc = msg;
if (!proc || (pd->start && (proc->start != pd->start)))
{
_proc_gone(pd);
return;
}
if (!strcmp(proc->state, "stop"))
{
elm_object_disabled_set(pd->btn_stop, EINA_TRUE);
elm_object_disabled_set(pd->btn_start, EINA_FALSE);
}
else
{
elm_object_disabled_set(pd->btn_stop, EINA_FALSE);
elm_object_disabled_set(pd->btn_start, EINA_TRUE);
}
elm_object_text_set(pd->entry_pid_cmd,
eina_slstr_printf("<subtitle>%s</subtitle>",
proc->command));
pwd_entry = getpwuid(proc->uid);
if (pwd_entry)
elm_object_text_set(pd->entry_pid_user, pwd_entry->pw_name);
if (proc->arguments)
elm_object_text_set(pd->entry_pid_cmd_args, proc->arguments);
else
elm_object_text_set(pd->entry_pid_cmd_args, "");
elm_object_text_set(pd->entry_pid_pid, eina_slstr_printf("%d", proc->pid));
elm_object_text_set(pd->entry_pid_uid, eina_slstr_printf("%d", proc->uid));
elm_object_text_set(pd->entry_pid_cpu,
eina_slstr_printf("%d", proc->cpu_id));
elm_object_text_set(pd->entry_pid_ppid, eina_slstr_printf("%d", proc->ppid));
elm_object_text_set(pd->entry_pid_threads,
eina_slstr_printf("%d", proc->numthreads));
elm_object_text_set(pd->entry_pid_virt, evisum_size_format(proc->mem_virt));
elm_object_text_set(pd->entry_pid_rss, evisum_size_format(proc->mem_rss));
#if !defined(__linux__)
elm_object_text_set(pd->entry_pid_shared, "N/A");
#else
elm_object_text_set(pd->entry_pid_shared,
evisum_size_format(proc->mem_shared));
#endif
elm_object_text_set(pd->entry_pid_size, evisum_size_format(proc->mem_size));
char *t = _time_string(proc->start);
if (t)
{
elm_object_text_set(pd->entry_pid_started, t);
free(t);
}
elm_object_text_set(pd->entry_pid_nice, eina_slstr_printf("%d", proc->nice));
elm_object_text_set(pd->entry_pid_pri,
eina_slstr_printf("%d", proc->priority));
elm_object_text_set(pd->entry_pid_state, proc->state);
if (pd->pid_cpu_time && proc->cpu_time >= pd->pid_cpu_time)
cpu_usage = (double)(proc->cpu_time - pd->pid_cpu_time) / pd->poll_delay;
elm_object_text_set(pd->entry_pid_cpu_usage,
eina_slstr_printf("%.1f%%", cpu_usage));
pd->pid_cpu_time = proc->cpu_time;
_thread_info_set(pd, proc);
proc_info_free(proc);
}
static void
_btn_start_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->selected_pid == -1)
return;
kill(pd->selected_pid, SIGCONT);
}
static void
_btn_stop_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->selected_pid == -1)
return;
kill(pd->selected_pid, SIGSTOP);
}
static void
_btn_kill_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->selected_pid == -1)
return;
kill(pd->selected_pid, SIGKILL);
}
static Evas_Object *
_entry_add(Evas_Object *parent)
{
Evas_Object *entry = elm_entry_add(parent);
evas_object_size_hint_weight_set(entry, EXPAND, EXPAND);
evas_object_size_hint_align_set(entry, FILL, FILL);
elm_entry_single_line_set(entry, EINA_TRUE);
elm_entry_scrollable_set(entry, EINA_TRUE);
elm_entry_editable_set(entry, EINA_FALSE);
elm_entry_line_wrap_set(entry, ELM_WRAP_NONE);
evas_object_show(entry);
return entry;
}
static Evas_Object *
_lb_add(Evas_Object *parent, const char *text)
{
Evas_Object *lb = elm_label_add(parent);
elm_object_text_set(lb, text);
evas_object_show(lb);
return lb;
}
static Evas_Object *
_process_tab_add(Evas_Object *parent, Ui_Data *pd)
{
Evas_Object *fr, *hbx, *tbl;
Evas_Object *lb, *entry, *btn, *pad, *ic;
Evas_Object *rec;
Proc_Info *proc;
int i = 0;
fr = elm_frame_add(parent);
elm_object_style_set(fr, "pad_small");
evas_object_size_hint_weight_set(fr, EXPAND, EXPAND);
evas_object_size_hint_align_set(fr, FILL, FILL);
tbl = elm_table_add(parent);
evas_object_size_hint_weight_set(tbl, EXPAND, EXPAND);
evas_object_size_hint_align_set(tbl, FILL, FILL);
evas_object_show(tbl);
elm_object_focus_allow_set(tbl, 1);
elm_object_content_set(fr, tbl);
rec = evas_object_rectangle_add(evas_object_evas_get(tbl));
evas_object_size_hint_min_set(rec, ELM_SCALE_SIZE(64), ELM_SCALE_SIZE(64));
evas_object_size_hint_max_set(rec, ELM_SCALE_SIZE(64), ELM_SCALE_SIZE(64));
evas_object_size_hint_align_set(rec, FILL, 1.0);
elm_table_pack(tbl, rec, 0, i, 1, 1);
proc = proc_info_by_pid(pd->selected_pid);
ic = elm_icon_add(parent);
evas_object_size_hint_weight_set(ic, EXPAND, EXPAND);
evas_object_size_hint_align_set(ic, FILL, FILL);
if (proc)
{
elm_icon_standard_set(ic,
evisum_icon_path_get(
evisum_icon_cache_find(proc)));
evas_object_show(ic);
proc_info_free(proc);
elm_table_pack(tbl, ic, 0, i, 1, 1);
}
lb = _lb_add(parent, _("Command:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_cmd = entry = elm_label_add(parent);
evas_object_size_hint_weight_set(entry, EXPAND, EXPAND);
evas_object_size_hint_align_set(entry, 0.0, 0.5);
evas_object_show(entry);
evas_object_hide(lb);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("Command line:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_cmd_args = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("PID:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_pid = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("Username:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_user = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("UID:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_uid = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("PPID:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_ppid = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
#if defined(__MacOS__)
lb = _lb_add(parent, _("WQ #:"));
#else
lb = _lb_add(parent, _("CPU #:"));
#endif
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_cpu = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("Threads:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_threads = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _(" Memory :"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_size = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _(" Shared memory:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_shared = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _(" Resident memory:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_rss = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _(" Virtual memory:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_virt = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _(" Start time:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_started = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("Nice:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_nice = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("Priority:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_pri = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("State:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_state = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
lb = _lb_add(parent, _("CPU %:"));
elm_table_pack(tbl, lb, 0, i, 1, 1);
pd->entry_pid_cpu_usage = entry = _entry_add(parent);
elm_table_pack(tbl, entry, 1, i++, 1, 1);
hbx = elm_box_add(parent);
evas_object_size_hint_weight_set(hbx, EXPAND, 0);
evas_object_size_hint_align_set(hbx, 1.0, FILL);
elm_box_horizontal_set(hbx, EINA_TRUE);
elm_box_homogeneous_set(hbx, EINA_TRUE);
evas_object_show(hbx);
elm_table_pack(tbl, hbx, 1, i, 1, 1);
pad = elm_frame_add(parent);
evas_object_size_hint_weight_set(pad, EXPAND, EXPAND);
elm_object_style_set(pad, "pad_small");
evas_object_show(pad);
elm_box_pack_end(hbx, pad);
pad = elm_frame_add(parent);
evas_object_size_hint_weight_set(pad, EXPAND, EXPAND);
elm_object_style_set(pad, "pad_small");
evas_object_show(pad);
elm_box_pack_end(hbx, pad);
btn = evisum_ui_button_add(parent, NULL, _("stop"), "stop",
_btn_stop_clicked_cb, pd);
evas_object_show(btn);
pd->btn_stop = btn;
elm_box_pack_end(hbx, btn);
btn = evisum_ui_button_add(parent, NULL, _("start"), "start",
_btn_start_clicked_cb, pd);
evas_object_show(btn);
pd->btn_start = btn;
elm_box_pack_end(hbx, btn);
btn = evisum_ui_button_add(parent, NULL, _("kill"), "kill",
_btn_kill_clicked_cb, pd);
evas_object_show(btn);
pd->btn_kill = btn;
elm_box_pack_end(hbx, btn);
return fr;
}
static void
_btn_icon_state_set(Evas_Object *btn, Eina_Bool reverse)
{
Evas_Object *icon = elm_icon_add(btn);
if (reverse)
elm_icon_standard_set(icon, evisum_icon_path_get("go-down"));
else
elm_icon_standard_set(icon, evisum_icon_path_get("go-up"));
elm_object_part_content_set(btn, "icon", icon);
evas_object_show(icon);
}
static void
_btn_name_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->sort_cb == _sort_by_name)
pd->sort_reverse = !pd->sort_reverse;
_btn_icon_state_set(obj, pd->sort_reverse);
pd->sort_cb = _sort_by_name;
}
static void
_tab_thread_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->sort_cb == _sort_by_tid)
pd->sort_reverse = !pd->sort_reverse;
_btn_icon_state_set(obj, pd->sort_reverse);
pd->sort_cb = _sort_by_tid;
}
static void
_btn_state_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->sort_cb == _sort_by_state)
pd->sort_reverse = !pd->sort_reverse;
_btn_icon_state_set(obj, pd->sort_reverse);
pd->sort_cb = _sort_by_state;
}
static void
_btn_cpu_id_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->sort_cb == _sort_by_cpu_id)
pd->sort_reverse = !pd->sort_reverse;
pd->sort_cb = _sort_by_cpu_id;
_btn_icon_state_set(obj, pd->sort_reverse);
}
static void
_btn_cpu_usage_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
if (pd->sort_cb == _sort_by_cpu_usage)
pd->sort_reverse = !pd->sort_reverse;
pd->sort_cb = _sort_by_cpu_usage;
_btn_icon_state_set(obj, pd->sort_reverse);
}
static Evas_Object *
_threads_tab_add(Evas_Object *parent, Ui_Data *pd)
{
Evas_Object *fr, *bx, *hbx, *btn, *genlist;
fr = elm_frame_add(parent);
evas_object_size_hint_weight_set(fr, EXPAND, EXPAND);
evas_object_size_hint_align_set(fr, FILL, FILL);
elm_object_style_set(fr, "pad_small");
bx = elm_box_add(parent);
evas_object_size_hint_weight_set(bx, EXPAND, EXPAND);
evas_object_size_hint_align_set(bx, FILL, FILL);
evas_object_show(bx);
elm_object_content_set(fr, bx);
hbx = elm_box_add(bx);
evas_object_size_hint_weight_set(hbx, EXPAND, 0);
evas_object_size_hint_align_set(hbx, FILL, FILL);
elm_box_horizontal_set(hbx, EINA_TRUE);
evas_object_show(hbx);
pd->tab_thread_id = btn = elm_button_add(hbx);
evas_object_size_hint_weight_set(btn, EXPAND, EXPAND);
evas_object_size_hint_align_set(btn, FILL, FILL);
elm_object_text_set(btn, _("id"));
_btn_icon_state_set(btn, pd->sort_reverse);
evas_object_show(btn);
evas_object_smart_callback_add(btn, "clicked", _tab_thread_clicked_cb, pd);
elm_box_pack_end(hbx, btn);
pd->tab_thread_name = btn = elm_button_add(hbx);
evas_object_size_hint_weight_set(btn, EXPAND, EXPAND);
evas_object_size_hint_align_set(btn, FILL, FILL);
elm_object_text_set(btn, _("name"));
_btn_icon_state_set(btn, pd->sort_reverse);
evas_object_show(btn);
evas_object_smart_callback_add(btn, "clicked", _btn_name_clicked_cb, pd);
elm_box_pack_end(hbx, btn);
pd->tab_thread_state = btn = elm_button_add(hbx);
evas_object_size_hint_weight_set(btn, EXPAND, EXPAND);
evas_object_size_hint_align_set(btn, FILL, FILL);
elm_object_text_set(btn, _("state"));
_btn_icon_state_set(btn, pd->sort_reverse);
evas_object_show(btn);
evas_object_smart_callback_add(btn, "clicked", _btn_state_clicked_cb, pd);
elm_box_pack_end(hbx, btn);
pd->tab_thread_cpu_id = btn = elm_button_add(hbx);
evas_object_size_hint_weight_set(btn, 0, EXPAND);
evas_object_size_hint_align_set(btn, FILL, FILL);
elm_object_text_set(btn, _("cpu id"));
_btn_icon_state_set(btn, pd->sort_reverse);
evas_object_show(btn);
evas_object_smart_callback_add(btn, "clicked", _btn_cpu_id_clicked_cb, pd);
elm_box_pack_end(hbx, btn);
pd->tab_thread_cpu_usage = btn = elm_button_add(hbx);
evas_object_size_hint_weight_set(btn, EXPAND, EXPAND);
evas_object_size_hint_align_set(btn, FILL, FILL);
elm_object_text_set(btn, _("cpu %"));
_btn_icon_state_set(btn, pd->sort_reverse);
evas_object_show(btn);
evas_object_smart_callback_add(btn, "clicked", _btn_cpu_usage_clicked_cb, pd);
elm_box_pack_end(hbx, btn);
pd->genlist_threads = genlist = elm_genlist_add(parent);
evas_object_data_set(genlist, "ui", pd);
elm_object_focus_allow_set(genlist, EINA_FALSE);
elm_genlist_homogeneous_set(genlist, EINA_TRUE);
elm_genlist_select_mode_set(genlist, ELM_OBJECT_SELECT_MODE_NONE);
evas_object_size_hint_weight_set(genlist, EXPAND, EXPAND);
evas_object_size_hint_align_set(genlist, FILL, FILL);
evas_object_show(genlist);
evas_object_smart_callback_add(pd->genlist_threads, "unrealized",
_item_unrealized_cb, pd);
elm_box_pack_end(bx, hbx);
elm_box_pack_end(bx, genlist);
return fr;
}
static Evas_Object *
_children_tab_add(Evas_Object *parent, Ui_Data *pd)
{
Evas_Object *fr, *bx, *genlist;
fr = elm_frame_add(parent);
evas_object_size_hint_weight_set(fr, EXPAND, EXPAND);
evas_object_size_hint_align_set(fr, FILL, FILL);
elm_object_style_set(fr, "pad_small");
bx = elm_box_add(parent);
evas_object_size_hint_weight_set(bx, EXPAND, EXPAND);
evas_object_size_hint_align_set(bx, FILL, FILL);
evas_object_show(bx);
elm_object_content_set(fr, bx);
pd->genlist_children = genlist = elm_genlist_add(parent);
evas_object_data_set(genlist, "ui", pd);
elm_object_focus_allow_set(genlist, EINA_FALSE);
elm_genlist_homogeneous_set(genlist, EINA_TRUE);
elm_genlist_select_mode_set(genlist, ELM_OBJECT_SELECT_MODE_DEFAULT);
evas_object_size_hint_weight_set(genlist, EXPAND, EXPAND);
evas_object_size_hint_align_set(genlist, FILL, FILL);
evas_object_show(genlist);
evas_object_smart_callback_add(genlist, "selected",
_item_children_clicked_cb, pd);
elm_box_pack_end(bx, genlist);
return fr;
}
static Evas_Object *
_manual_tab_add(Evas_Object *parent, Ui_Data *pd)
{
Evas_Object *fr, *bx, *entry;
Evas_Object *tb;
int sz;
fr = elm_frame_add(parent);
evas_object_size_hint_weight_set(fr, EXPAND, EXPAND);
evas_object_size_hint_align_set(fr, FILL, FILL);
elm_object_style_set(fr, "pad_small");
bx = elm_box_add(parent);
evas_object_size_hint_weight_set(bx, EXPAND, EXPAND);
evas_object_size_hint_align_set(bx, FILL, FILL);
evas_object_show(bx);
elm_object_content_set(fr, bx);
pd->entry_manual = entry = elm_entry_add(bx);
evas_object_size_hint_weight_set(entry, EXPAND, EXPAND);
evas_object_size_hint_align_set(entry, FILL, FILL);
elm_entry_single_line_set(entry, EINA_FALSE);
elm_entry_line_wrap_set(entry, ELM_WRAP_NONE);
elm_entry_editable_set(entry, EINA_FALSE);
elm_entry_scrollable_set(entry, EINA_TRUE);
evas_object_show(entry);
elm_box_pack_end(bx, entry);
tb = elm_entry_textblock_get(entry);
sz = evisum_ui_textblock_font_size_get(tb);
evisum_ui_textblock_font_size_set(tb, sz - 1);
return fr;
}
static void
_tab_change(Ui_Data *pd, Evas_Object *view)
{
Elm_Transit *trans;
elm_object_disabled_set(pd->tab_main, EINA_FALSE);
elm_object_disabled_set(pd->tab_manual, EINA_FALSE);
elm_object_disabled_set(pd->tab_thread, EINA_FALSE);
elm_object_disabled_set(pd->tab_children, EINA_FALSE);
evas_object_hide(pd->main_view);
evas_object_hide(pd->children_view);
evas_object_hide(pd->manual_view);
evas_object_hide(pd->thread_view);
trans = elm_transit_add();
elm_transit_object_add(trans, pd->current_view);
elm_transit_object_add(trans, view);
elm_transit_duration_set(trans, 0.15);
elm_transit_effect_blend_add(trans);
pd->current_view = view;
evas_object_show(view);
elm_transit_go(trans);
}
static void
_tab_main_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
_tab_change(pd, pd->main_view);
elm_object_disabled_set(obj, 1);
}
static void
_tab_children_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
_children_view_update(pd);
_tab_change(pd, pd->children_view);
elm_object_disabled_set(obj, 1);
}
static void
_tab_threads_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd = data;
_tab_change(pd, pd->thread_view);
elm_object_disabled_set(obj, 1);
}
static void
_tab_manual_clicked_cb(void *data, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Ui_Data *pd;
Eina_List *lines = NULL;
pd = data;
_tab_change(pd, pd->manual_view);
elm_object_disabled_set(obj, 1);
if (pd->manual_init) return;
setenv("MANWIDTH", "80", 1);
if (pd->selected_cmd && pd->selected_cmd[0] && !strchr(pd->selected_cmd, ' '))
lines =_exe_response(eina_slstr_printf("man %s | col -bx", pd->selected_cmd));
if (!lines)
{
if (!strcmp(pd->selected_cmd, "evisum"))
elm_object_text_set(pd->entry_manual, _evisum_docs());
else
{
elm_object_text_set(pd->entry_manual,
eina_slstr_printf(
_("No documentation found for %s."),
pd->selected_cmd));
}
}
else
{
char *line;
Evas_Object *ent = pd->entry_manual;
int n = 1;
elm_entry_entry_append(ent, "<code>");
EINA_LIST_FREE(lines, line)
{
if (n++ > 1)
elm_entry_entry_append(ent, eina_slstr_printf("%s<br>", line));
free(line);
}
elm_entry_entry_append(ent, "</code>");
}
unsetenv("MANWIDTH");
pd->manual_init = EINA_TRUE;
}
static Evas_Object *
_tabs_add(Evas_Object *parent, Ui_Data *pd)
{
Evas_Object *hbx, *pad, *btn;
evisum_ui_background_random_add(parent,
evisum_ui_backgrounds_enabled_get());
hbx = elm_box_add(parent);
evas_object_size_hint_weight_set(hbx, EXPAND, 0);
evas_object_size_hint_align_set(hbx, FILL, 0.5);
elm_box_horizontal_set(hbx, EINA_TRUE);
evas_object_show(hbx);
pad = elm_frame_add(parent);
elm_object_style_set(pad, "pad_medium");
evas_object_size_hint_weight_set(pad, EXPAND, EXPAND);
evas_object_size_hint_align_set(pad, FILL, FILL);
evas_object_show(pad);
elm_box_pack_end(hbx, pad);
pad = elm_frame_add(parent);
elm_object_style_set(pad, "pad_small");
evas_object_size_hint_weight_set(pad, 0.0, EXPAND);
evas_object_size_hint_align_set(pad, FILL, FILL);
evas_object_show(pad);
btn = evisum_ui_tab_add(parent, &pd->tab_main, _("Process"),
_tab_main_clicked_cb, pd);
elm_object_disabled_set(pd->tab_main, EINA_TRUE);
elm_object_content_set(pad, btn);
elm_box_pack_end(hbx, pad);
pad = elm_frame_add(parent);
elm_object_style_set(pad, "pad_small");
evas_object_size_hint_weight_set(pad, 0.0, EXPAND);
evas_object_size_hint_align_set(pad, FILL, FILL);
evas_object_show(pad);
btn = evisum_ui_tab_add(parent, &pd->tab_children, _("Children"),
_tab_children_clicked_cb, pd);
elm_object_content_set(pad, btn);
elm_box_pack_end(hbx, pad);
pad = elm_frame_add(parent);
elm_object_style_set(pad, "pad_small");
evas_object_size_hint_weight_set(pad, 0.0, EXPAND);
evas_object_size_hint_align_set(pad, FILL, FILL);
evas_object_show(pad);
btn = evisum_ui_tab_add(parent, &pd->tab_thread, _("Threads"),
_tab_threads_clicked_cb, pd);
elm_object_content_set(pad, btn);
elm_box_pack_end(hbx, pad);
pad = elm_frame_add(parent);
elm_object_style_set(pad, "pad_small");
evas_object_size_hint_weight_set(pad, 0.0, EXPAND);
evas_object_size_hint_align_set(pad, FILL, FILL);
evas_object_show(pad);
btn = evisum_ui_tab_add(parent, &pd->tab_manual, _("Manual"),
_tab_manual_clicked_cb, pd);
elm_object_content_set(pad, btn);
elm_box_pack_end(hbx, pad);
pad = elm_frame_add(parent);
elm_object_style_set(pad, "pad_medium");
evas_object_size_hint_weight_set(pad, EXPAND, EXPAND);
evas_object_size_hint_align_set(pad, FILL, FILL);
evas_object_show(pad);
elm_box_pack_end(hbx, pad);
return hbx;
}
static void
_win_del_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED,
void *event_info EINA_UNUSED)
{
Evas_Object *win;
Ui_Data *pd;
pd = data;
win = obj;
if (pd->hash_cpu_times)
eina_hash_free(pd->hash_cpu_times);
if (pd->thread)
{
ecore_thread_cancel(pd->thread);
ecore_thread_wait(pd->thread, 0.5);
}
if (pd->selected_cmd)
free(pd->selected_cmd);
if (pd->cache)
evisum_ui_item_cache_free(pd->cache);
evas_object_del(win);
free(pd);
}
static void
_win_resize_cb(void *data, Evas *e, Evas_Object *obj, void *event_info)
{
Ui_Data *pd = data;
elm_genlist_realized_items_update(pd->genlist_threads);
}
static void
_win_key_down_cb(void *data, Evas *e, Evas_Object *obj, void *event_info)
{
Evas_Event_Key_Down *ev;
Ui_Data *pd;
pd = data;
ev = event_info;
if (!ev || !ev->keyname)
return;
if (!strcmp(ev->keyname, "Escape"))
evas_object_del(pd->win);
}
static void
_action_do(Ui_Data *pd, Evisum_Proc_Action action)
{
switch (action)
{
case PROC_VIEW_DEFAULT:
pd->current_view = pd->main_view;
_tab_main_clicked_cb(pd, pd->tab_main, NULL);
break;
case PROC_VIEW_CHILDREN:
pd->current_view = pd->children_view;
_tab_children_clicked_cb(pd, pd->tab_children, NULL);
break;
case PROC_VIEW_THREADS:
pd->current_view = pd->thread_view;
_tab_threads_clicked_cb(pd, pd->tab_thread, NULL);
break;
case PROC_VIEW_MANUAL:
pd->current_view = pd->manual_view;
_tab_manual_clicked_cb(pd, pd->tab_manual, NULL);
break;
}
evas_object_show(pd->current_view);
}
void
ui_process_view_win_add(int pid, Evisum_Proc_Action action)
{
Evas_Object *win, *ic, *bx, *tabs;
Proc_Info *proc;
Ui_Data *pd = calloc(1, sizeof(Ui_Data));
pd->selected_pid = pid;
pd->poll_delay = 1;
pd->cache = NULL;
pd->sort_reverse = EINA_TRUE;
pd->sort_cb = _sort_by_cpu_usage;
proc = proc_info_by_pid(pid);
if (!proc)
pd->selected_cmd = strdup(_("Unknown"));
else
{
pd->selected_cmd = strdup(proc->command);
pd->start = proc->start;
proc_info_free(proc);
}
pd->win = win = elm_win_util_standard_add("evisum", "evisum");
elm_win_autodel_set(win, EINA_TRUE);
ic = elm_icon_add(win);
elm_icon_standard_set(ic, "evisum");
elm_win_icon_object_set(win, ic);
elm_win_title_set(pd->win, eina_slstr_printf("%s (%i)",
pd->selected_cmd, pd->selected_pid));
tabs = _tabs_add(win, pd);
bx = elm_box_add(win);
evas_object_size_hint_weight_set(bx, EXPAND, 0);
evas_object_size_hint_align_set(bx, FILL, FILL);
evas_object_show(bx);
elm_box_pack_end(bx, tabs);
pd->content = elm_table_add(bx);
evas_object_size_hint_weight_set(pd->content, 0.5, EXPAND);
evas_object_size_hint_align_set(pd->content, FILL, 0.0);
evas_object_show(pd->content);
pd->main_view = _process_tab_add(tabs, pd);
pd->children_view = _children_tab_add(tabs, pd);
pd->thread_view = _threads_tab_add(tabs, pd);
pd->manual_view = _manual_tab_add(tabs, pd);
elm_table_pack(pd->content, pd->manual_view, 0, 0, 1, 1);
elm_table_pack(pd->content, pd->children_view, 0, 0, 1, 1);
elm_table_pack(pd->content, pd->main_view, 0, 0, 1, 1);
elm_table_pack(pd->content, pd->thread_view, 0, 0, 1, 1);
elm_box_pack_end(bx, pd->content);
elm_object_content_set(win, bx);
evas_object_event_callback_add(win, EVAS_CALLBACK_DEL, _win_del_cb, pd);
evas_object_event_callback_add(win, EVAS_CALLBACK_RESIZE, _win_resize_cb, pd);
evas_object_event_callback_add(bx, EVAS_CALLBACK_KEY_DOWN, _win_key_down_cb, pd);
evas_object_resize(win, 480 * elm_config_scale_get(), -1);
elm_win_center(win, 1, 1);
evas_object_show(win);
_action_do(pd, action);
pd->cache = evisum_ui_item_cache_new(pd->genlist_threads, _item_create, 10);
pd->thread = ecore_thread_feedback_run(_proc_info_main,
_proc_info_feedback_cb,
NULL,
NULL,
pd, EINA_FALSE);
}