Skip to content

Commit

Permalink
perf tools: Add general function to parse sublevel options
Browse files Browse the repository at this point in the history
This factors out a general function perf_parse_sublevel_options() to
parse sublevel options. The 'sublevel' options is something like the
'--debug' options which allow more sublevel options.

Signed-off-by: Changbin Du <[email protected]>
Acked-by: Namhyung Kim <[email protected]>
Cc: Jiri Olsa <[email protected]>
Cc: Peter Zijlstra <[email protected]>
Cc: Steven Rostedt (VMware) <[email protected]>
Link: http://lore.kernel.org/lkml/[email protected]
Signed-off-by: Arnaldo Carvalho de Melo <[email protected]>
  • Loading branch information
changbindu authored and acmel committed Aug 14, 2020
1 parent 5b34747 commit a80abe2
Show file tree
Hide file tree
Showing 4 changed files with 99 additions and 44 deletions.
1 change: 1 addition & 0 deletions tools/perf/util/Build
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,7 @@ endif
perf-y += parse-branch-options.o
perf-y += dump-insn.o
perf-y += parse-regs-options.o
perf-y += parse-sublevel-options.o
perf-y += term.o
perf-y += help-unknown-cmd.o
perf-y += mem-events.o
Expand Down
61 changes: 17 additions & 44 deletions tools/perf/util/debug.c
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
#include "target.h"
#include "ui/helpline.h"
#include "ui/ui.h"
#include "util/parse-sublevel-options.h"

#include <linux/ctype.h>

Expand Down Expand Up @@ -173,65 +174,37 @@ void trace_event(union perf_event *event)
trace_event_printer, event);
}

static struct debug_variable {
const char *name;
int *ptr;
} debug_variables[] = {
{ .name = "verbose", .ptr = &verbose },
{ .name = "ordered-events", .ptr = &debug_ordered_events},
{ .name = "stderr", .ptr = &redirect_to_stderr},
{ .name = "data-convert", .ptr = &debug_data_convert },
{ .name = "perf-event-open", .ptr = &debug_peo_args },
static struct sublevel_option debug_opts[] = {
{ .name = "verbose", .value_ptr = &verbose },
{ .name = "ordered-events", .value_ptr = &debug_ordered_events},
{ .name = "stderr", .value_ptr = &redirect_to_stderr},
{ .name = "data-convert", .value_ptr = &debug_data_convert },
{ .name = "perf-event-open", .value_ptr = &debug_peo_args },
{ .name = NULL, }
};

int perf_debug_option(const char *str)
{
struct debug_variable *var = &debug_variables[0];
char *vstr, *s = strdup(str);
int v = 1;

vstr = strchr(s, '=');
if (vstr)
*vstr++ = 0;

while (var->name) {
if (!strcmp(s, var->name))
break;
var++;
}

if (!var->name) {
pr_err("Unknown debug variable name '%s'\n", s);
free(s);
return -1;
}
int ret;

if (vstr) {
v = atoi(vstr);
/*
* Allow only values in range (0, 10),
* otherwise set 0.
*/
v = (v < 0) || (v > 10) ? 0 : v;
}
ret = perf_parse_sublevel_options(str, debug_opts);
if (ret)
return ret;

if (quiet)
v = -1;
/* Allow only verbose value in range (0, 10), otherwise set 0. */
verbose = (verbose < 0) || (verbose > 10) ? 0 : verbose;

*var->ptr = v;
free(s);
return 0;
}

int perf_quiet_option(void)
{
struct debug_variable *var = &debug_variables[0];
struct sublevel_option *opt = &debug_opts[0];

/* disable all debug messages */
while (var->name) {
*var->ptr = -1;
var++;
while (opt->name) {
*opt->value_ptr = -1;
opt++;
}

return 0;
Expand Down
70 changes: 70 additions & 0 deletions tools/perf/util/parse-sublevel-options.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
#include <stdlib.h>
#include <stdint.h>
#include <string.h>
#include <stdio.h>

#include "util/debug.h"
#include "util/parse-sublevel-options.h"

static int parse_one_sublevel_option(const char *str,
struct sublevel_option *opts)
{
struct sublevel_option *opt = opts;
char *vstr, *s = strdup(str);
int v = 1;

if (!s) {
pr_err("no memory\n");
return -1;
}

vstr = strchr(s, '=');
if (vstr)
*vstr++ = 0;

while (opt->name) {
if (!strcmp(s, opt->name))
break;
opt++;
}

if (!opt->name) {
pr_err("Unknown option name '%s'\n", s);
free(s);
return -1;
}

if (vstr)
v = atoi(vstr);

*opt->value_ptr = v;
free(s);
return 0;
}

/* parse options like --foo a=<n>,b,c... */
int perf_parse_sublevel_options(const char *str, struct sublevel_option *opts)
{
char *s = strdup(str);
char *p = NULL;
int ret;

if (!s) {
pr_err("no memory\n");
return -1;
}

p = strtok(s, ",");
while (p) {
ret = parse_one_sublevel_option(p, opts);
if (ret) {
free(s);
return ret;
}

p = strtok(NULL, ",");
}

free(s);
return 0;
}
11 changes: 11 additions & 0 deletions tools/perf/util/parse-sublevel-options.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
#ifndef _PERF_PARSE_SUBLEVEL_OPTIONS_H
#define _PERF_PARSE_SUBLEVEL_OPTIONS_H

struct sublevel_option {
const char *name;
int *value_ptr;
};

int perf_parse_sublevel_options(const char *str, struct sublevel_option *opts);

#endif

0 comments on commit a80abe2

Please sign in to comment.