top: 'other filters' saved with config file, pgm logic

Well, after the rearranging and refactoring, all those
active 'other filter' entries for each window will now
be preserved in the user's configuration file via 'W'.

For raising the issue below, thanks to Marco Ippolito.

Reference(s):
https://gitlab.com/procps-ng/procps/issues/99

Signed-off-by: Jim Warner <james.warner@comcast.net>
This commit is contained in:
Jim Warner 2018-06-17 00:00:00 -05:00 committed by Craig Small
parent 993d22211f
commit a6f86732f2
3 changed files with 140 additions and 35 deletions

169
top/top.c
View File

@ -229,6 +229,14 @@ static float Graph_adj; // bars/blocks scaling factor
static int Graph_len; // scaled length (<= GRAPH_actual) static int Graph_len; // scaled length (<= GRAPH_actual)
static const char Graph_blks[] = " "; static const char Graph_blks[] = " ";
static const char Graph_bars[] = "||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||"; static const char Graph_bars[] = "||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||";
/* Support for 'Other Filters' in the configuration file */
static const char Osel_delim_1_txt[] = "begin: saved other filter data -------------------\n";
static const char Osel_delim_2_txt[] = "end : saved other filter data -------------------\n";
static const char Osel_window_fmts[] = "window #%d, osel_tot=%d\n";
#define OSEL_FILTER "filter="
static const char Osel_filterO_fmt[] = "\ttype=%d,\t" OSEL_FILTER "%s\n";
static const char Osel_filterI_fmt[] = "\ttype=%d,\t" OSEL_FILTER "%*s\n";
/*###### Sort callbacks ################################################*/ /*###### Sort callbacks ################################################*/
@ -3482,11 +3490,12 @@ struct osel_s {
int ops; // filter delimiter/operation int ops; // filter delimiter/operation
int inc; // include == 1, exclude == 0 int inc; // include == 1, exclude == 0
int enu; // field (procflag) to filter int enu; // field (procflag) to filter
int typ; // typ used to set: rel & sel
}; };
/* /*
* A function to parse, validate and build a single 'other filter' */ * A function to parse, validate and build a single 'other filter' */
static const char *osel_add (int ch, char *glob) { static const char *osel_add (WIN_t *q, int ch, char *glob, int push) {
int (*rel)(const char *, const char *); int (*rel)(const char *, const char *);
char *(*sel)(const char *, const char *); char *(*sel)(const char *, const char *);
char raw[MEDBUFSIZ], ops, *pval; char raw[MEDBUFSIZ], ops, *pval;
@ -3503,7 +3512,7 @@ static const char *osel_add (int ch, char *glob) {
if (!snprintf(raw, sizeof(raw), "%s", glob)) if (!snprintf(raw, sizeof(raw), "%s", glob))
return NULL; return NULL;
for (osel = Curwin->osel_1st; osel; ) { for (osel = q->osel_1st; osel; ) {
if (!strcmp(osel->raw, raw)) // #1: is criteria duplicate? if (!strcmp(osel->raw, raw)) // #1: is criteria duplicate?
return N_txt(OSEL_errdups_txt); return N_txt(OSEL_errdups_txt);
osel = osel->nxt; osel = osel->nxt;
@ -3525,10 +3534,9 @@ static const char *osel_add (int ch, char *glob) {
if (!(*pval)) // #5: did we get some value? if (!(*pval)) // #5: did we get some value?
return fmtmk(N_fmt(OSEL_errvalu_fmt) return fmtmk(N_fmt(OSEL_errvalu_fmt)
, inc ? N_txt(WORD_include_txt) : N_txt(WORD_exclude_txt)); , inc ? N_txt(WORD_include_txt) : N_txt(WORD_exclude_txt));
if (Curwin->osel_prt && strlen(Curwin->osel_prt) >= INT_MAX - (sizeof(raw) + 6))
return NULL;
osel = alloc_c(sizeof(struct osel_s)); osel = alloc_c(sizeof(struct osel_s));
osel->typ = ch;
osel->inc = inc; osel->inc = inc;
osel->enu = enu; osel->enu = enu;
osel->ops = ops; osel->ops = ops;
@ -3538,13 +3546,24 @@ static const char *osel_add (int ch, char *glob) {
osel->sel = sel; osel->sel = sel;
osel->raw = alloc_s(raw); osel->raw = alloc_s(raw);
osel->nxt = Curwin->osel_1st; if (push) {
Curwin->osel_1st = osel; // a LIFO queue was used when we're interactive
Curwin->osel_tot += 1; osel->nxt = q->osel_1st;
q->osel_1st = osel;
if (!Curwin->osel_prt) Curwin->osel_prt = alloc_c(strlen(raw) + 3); } else {
else Curwin->osel_prt = alloc_r(Curwin->osel_prt, strlen(Curwin->osel_prt) + strlen(raw) + 6); // a FIFO queue must be employed for the rcfile
strcat(Curwin->osel_prt, fmtmk("%s'%s'", (Curwin->osel_tot > 1) ? " + " : "", raw)); if (!q->osel_1st)
q->osel_1st = osel;
else {
struct osel_s *prev, *walk = q->osel_1st;
do {
prev = walk;
walk = walk->nxt;
} while (walk);
prev->nxt = osel;
}
}
q->osel_tot += 1;
return NULL; return NULL;
} // end: osel_add } // end: osel_add
@ -3564,10 +3583,8 @@ static void osel_clear (WIN_t *q) {
} }
q->osel_tot = 0; q->osel_tot = 0;
q->osel_1st = NULL; q->osel_1st = NULL;
free (q->osel_prt);
q->osel_prt = NULL;
#ifndef USE_X_COLHDR #ifndef USE_X_COLHDR
OFFw(Curwin, NOHISEL_xxx); OFFw(q, NOHISEL_xxx);
#endif #endif
} // end: osel_clear } // end: osel_clear
@ -3849,6 +3866,50 @@ static int config_insp (FILE *fp, char *buf, size_t size) {
} // end: config_insp } // end: config_insp
/*
* A configs_file *Helper* function responsible for reading
* and validating a configuration file's 'Other Filter' entries */
static int config_osel (FILE *fp, char *buf, size_t size) {
int i, ch, tot, wno, begun;
char *p;
for (begun = 0;;) {
if (!fgets(buf, size, fp)) return 0;
if (buf[0] == '\n') continue;
// whoa, must be an 'inspect' entry
if (!begun && !strstr(buf, Osel_delim_1_txt))
return 0;
// ok, we're now begining
if (!begun && strstr(buf, Osel_delim_1_txt)) {
begun = 1;
continue;
}
// this marks the end of our stuff
if (begun && strstr(buf, Osel_delim_2_txt))
break;
if (2 != sscanf(buf, Osel_window_fmts, &wno, &tot))
goto end_oops;
for (i = 0; i < tot; i++) {
if (!fgets(buf, size, fp)) return 1;
if (1 > sscanf(buf, Osel_filterI_fmt, &ch)) goto end_oops;
if ((p = strchr(buf, '\n'))) *p = '\0';
if (!(p = strstr(buf, OSEL_FILTER))) goto end_oops;
p += sizeof(OSEL_FILTER) - 1;
if (osel_add(&Winstk[wno], ch, p, 0)) goto end_oops;
}
}
// let's prime that buf for the next guy...
fgets(buf, size, fp);
return 0;
end_oops:
Rc_questions = 1;
return 1;
} // end: config_osel
/* /*
* A configs_reads *Helper* function responsible for processing * A configs_reads *Helper* function responsible for processing
* a configuration file (personal or system-wide default) */ * a configuration file (personal or system-wide default) */
@ -3960,7 +4021,9 @@ static const char *configs_file (FILE *fp, const char *name, float *delay) {
Rc.zero_suppress = 0; Rc.zero_suppress = 0;
// lastly, let's process any optional glob(s) ... // lastly, let's process any optional glob(s) ...
// (darn, must do osel 1st even though alphabetically 2nd)
fbuf[0] = '\0'; fbuf[0] = '\0';
config_osel(fp, fbuf, sizeof(fbuf));
config_insp(fp, fbuf, sizeof(fbuf)); config_insp(fp, fbuf, sizeof(fbuf));
return NULL; return NULL;
@ -3999,7 +4062,8 @@ static int configs_path (const char *const fmts, ...) {
* line b: contains w->winflags, sortindx, maxtasks, graph modes * line b: contains w->winflags, sortindx, maxtasks, graph modes
* line c: contains w->summclr, msgsclr, headclr, taskclr * line c: contains w->summclr, msgsclr, headclr, taskclr
* line 15 : miscellaneous additional global settings * line 15 : miscellaneous additional global settings
* Any remaining lines are devoted to the 'Inspect Other' feature * Any remaining lines are devoted to the optional entries
* supporting the 'Other Filter' and 'Inspect' provisions.
* 3. 'SYS_RCDEFAULTS' system-wide defaults if 'Rc_name' absent * 3. 'SYS_RCDEFAULTS' system-wide defaults if 'Rc_name' absent
* format is identical to #2 above */ * format is identical to #2 above */
static void configs_reads (void) { static void configs_reads (void) {
@ -4728,23 +4792,51 @@ signify_that:
static void other_filters (int ch) { static void other_filters (int ch) {
WIN_t *w = Curwin; // avoid gcc bloat with a local copy
const char *txt, *p; const char *txt, *p;
char *glob; char *glob;
if (ch == 'o') txt = N_txt(OSEL_casenot_txt); switch (ch) {
else txt = N_txt(OSEL_caseyes_txt); case 'o':
case 'O':
glob = ioline(fmtmk(N_fmt(OSEL_prompts_fmt), Curwin->osel_tot + 1, txt)); if (ch == 'o') txt = N_txt(OSEL_casenot_txt);
if (*glob == kbd_ESC || !*glob) else txt = N_txt(OSEL_caseyes_txt);
return; glob = ioline(fmtmk(N_fmt(OSEL_prompts_fmt), w->osel_tot + 1, txt));
if (*glob == kbd_ESC || *glob == '\0')
if ((p = osel_add(ch, glob))) { return;
show_msg(p); if ((p = osel_add(w, ch, glob, 1))) {
return; show_msg(p);
} return;
}
#ifndef USE_X_COLHDR #ifndef USE_X_COLHDR
SETw(Curwin, NOHISEL_xxx); SETw(w, NOHISEL_xxx);
#endif #endif
break;
case kbd_CtrlO:
if (VIZCHKw(w)) {
char buf[SCREENMAX], **pp;
struct osel_s *osel;
int i;
i = 0;
osel = w->osel_1st;
pp = alloc_c((w->osel_tot + 1) * sizeof(char**));
while (osel && i < w->osel_tot) {
pp[i++] = osel->raw;
osel = osel->nxt;
}
buf[0] = '\0';
for ( ; i > 0; )
strncat(buf, fmtmk("%s'%s'", " + " , pp[--i]), sizeof(buf) - (strlen(buf) + 1));
if (buf[0]) p = buf + strspn(buf, " + ");
else p = N_txt(WORD_noneone_txt);
ioline(fmtmk(N_fmt(OSEL_statlin_fmt), p));
free(pp);
}
break;
default: // keep gcc happy
break;
}
} // end: other_filters } // end: other_filters
@ -4785,6 +4877,23 @@ static void write_rcfile (void) {
fprintf(fp, "Fixed_widest=%d, Summ_mscale=%d, Task_mscale=%d, Zero_suppress=%d\n" fprintf(fp, "Fixed_widest=%d, Summ_mscale=%d, Task_mscale=%d, Zero_suppress=%d\n"
, Rc.fixed_widest, Rc.summ_mscale, Rc.task_mscale, Rc.zero_suppress); , Rc.fixed_widest, Rc.summ_mscale, Rc.task_mscale, Rc.zero_suppress);
if (Winstk[0].osel_tot + Winstk[1].osel_tot
+ Winstk[2].osel_tot + Winstk[3].osel_tot) {
fprintf(fp, "\n");
fprintf(fp, Osel_delim_1_txt);
for (i = 0 ; i < GROUPSMAX; i++) {
struct osel_s *osel = Winstk[i].osel_1st;
if (osel) {
fprintf(fp, Osel_window_fmts, i, Winstk[i].osel_tot);
do {
fprintf(fp, Osel_filterO_fmt, osel->typ, osel->raw);
osel = osel->nxt;
} while (osel);
}
}
fprintf(fp, Osel_delim_2_txt);
}
if (Inspect.raw) if (Inspect.raw)
fputs(Inspect.raw, fp); fputs(Inspect.raw, fp);
@ -5084,6 +5193,7 @@ static void keys_task (int ch) {
break; break;
case 'O': case 'O':
case 'o': case 'o':
case kbd_CtrlO:
if (VIZCHKw(w)) other_filters(ch); if (VIZCHKw(w)) other_filters(ch);
break; break;
case 'U': case 'U':
@ -5134,11 +5244,6 @@ static void keys_task (int ch) {
capsmk(w); capsmk(w);
} }
break; break;
case kbd_CtrlO:
if (VIZCHKw(w))
ioline(fmtmk(N_fmt(OSEL_statlin_fmt)
, w->osel_prt ? w->osel_prt : N_txt(WORD_noneone_txt)));
break;
default: // keep gcc happy default: // keep gcc happy
break; break;
} }

View File

@ -432,7 +432,6 @@ typedef struct WIN_t {
*captab [CAPTABMAX]; // captab needed by show_special() *captab [CAPTABMAX]; // captab needed by show_special()
struct osel_s *osel_1st; // other selection criteria anchor struct osel_s *osel_1st; // other selection criteria anchor
int osel_tot; // total of other selection criteria int osel_tot; // total of other selection criteria
char *osel_prt; // other stuff printable as status line
char *findstr; // window's current/active search string char *findstr; // window's current/active search string
int findlen; // above's strlen, without call overhead int findlen; // above's strlen, without call overhead
proc_t **ppt; // this window's proc_t ptr array proc_t **ppt; // this window's proc_t ptr array
@ -757,13 +756,14 @@ typedef struct WIN_t {
//atic int insp_view_choice (proc_t *obj); //atic int insp_view_choice (proc_t *obj);
//atic void inspection_utility (int pid); //atic void inspection_utility (int pid);
/*------ Other Filtering ------------------------------------------------*/ /*------ Other Filtering ------------------------------------------------*/
//atic const char *osel_add (int ch, char *glob); //atic const char *osel_add (WIN_t *q, int ch, char *glob, int push);
//atic void osel_clear (WIN_t *q); //atic void osel_clear (WIN_t *q);
//atic inline int osel_matched (const WIN_t *q, FLG_t enu, const char *str); //atic inline int osel_matched (const WIN_t *q, FLG_t enu, const char *str);
/*------ Startup routines ----------------------------------------------*/ /*------ Startup routines ----------------------------------------------*/
//atic void before (char *me); //atic void before (char *me);
//atic int config_cvt (WIN_t *q); //atic int config_cvt (WIN_t *q);
//atic int config_insp (FILE *fp, char *buf, size_t size); //atic int config_insp (FILE *fp, char *buf, size_t size);
//atic int config_osel (FILE *fp, char *buf, size_t size);
//atic const char *configs_file (FILE *fp, const char *name, float *delay); //atic const char *configs_file (FILE *fp, const char *name, float *delay);
//atic int configs_path (const char *const fmts, ...); //atic int configs_path (const char *const fmts, ...);
//atic void configs_reads (void); //atic void configs_reads (void);

View File

@ -424,7 +424,7 @@ static void build_norm_nlstab (void) {
Norm_nlstab[GET_find_str_txt] = _("Locate string"); Norm_nlstab[GET_find_str_txt] = _("Locate string");
Norm_nlstab[FIND_no_find_fmt] = _("%s\"%s\" not found"); Norm_nlstab[FIND_no_find_fmt] = _("%s\"%s\" not found");
Norm_nlstab[XTRA_fixwide_fmt] = _("width incr is %d, change to (0 default, -1 auto)"); Norm_nlstab[XTRA_fixwide_fmt] = _("width incr is %d, change to (0 default, -1 auto)");
Norm_nlstab[XTRA_warncfg_txt] = _("rcfile has 'inspect' entry error(s), write anyway?"); Norm_nlstab[XTRA_warncfg_txt] = _("rcfile has inspect/other-filter error(s), save anyway?");
Norm_nlstab[XTRA_badflds_fmt] = _("unrecognized field name '%s'"); Norm_nlstab[XTRA_badflds_fmt] = _("unrecognized field name '%s'");
Norm_nlstab[XTRA_winsize_txt] = _("even using field names only, window is now too small"); Norm_nlstab[XTRA_winsize_txt] = _("even using field names only, window is now too small");
#ifndef INSP_OFFDEMO #ifndef INSP_OFFDEMO