diff options
author | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2013-03-17 22:40:50 -0400 |
---|---|---|
committer | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2013-03-17 22:40:50 -0400 |
commit | 688d794c4c3f8b08c814381ee2edd3ede5856056 (patch) | |
tree | ef680add71e2a9588d07d8b594edbc1b5cd127d7 /tools/perf/ui/browsers/scripts.c | |
parent | 16142655269aaf580488e074eabfdcf0fb4e3687 (diff) | |
parent | a937536b868b8369b98967929045f1df54234323 (diff) |
Merge tag 'v3.9-rc3' into next
Merge with mainline to bring in module_platform_driver_probe() and
devm_ioremap_resource().
Diffstat (limited to 'tools/perf/ui/browsers/scripts.c')
-rw-r--r-- | tools/perf/ui/browsers/scripts.c | 189 |
1 files changed, 189 insertions, 0 deletions
diff --git a/tools/perf/ui/browsers/scripts.c b/tools/perf/ui/browsers/scripts.c new file mode 100644 index 000000000000..cbbd44b0d93e --- /dev/null +++ b/tools/perf/ui/browsers/scripts.c | |||
@@ -0,0 +1,189 @@ | |||
1 | #include <elf.h> | ||
2 | #include <newt.h> | ||
3 | #include <inttypes.h> | ||
4 | #include <sys/ttydefaults.h> | ||
5 | #include <string.h> | ||
6 | #include "../../util/sort.h" | ||
7 | #include "../../util/util.h" | ||
8 | #include "../../util/hist.h" | ||
9 | #include "../../util/debug.h" | ||
10 | #include "../../util/symbol.h" | ||
11 | #include "../browser.h" | ||
12 | #include "../helpline.h" | ||
13 | #include "../libslang.h" | ||
14 | |||
15 | /* 2048 lines should be enough for a script output */ | ||
16 | #define MAX_LINES 2048 | ||
17 | |||
18 | /* 160 bytes for one output line */ | ||
19 | #define AVERAGE_LINE_LEN 160 | ||
20 | |||
21 | struct script_line { | ||
22 | struct list_head node; | ||
23 | char line[AVERAGE_LINE_LEN]; | ||
24 | }; | ||
25 | |||
26 | struct perf_script_browser { | ||
27 | struct ui_browser b; | ||
28 | struct list_head entries; | ||
29 | const char *script_name; | ||
30 | int nr_lines; | ||
31 | }; | ||
32 | |||
33 | #define SCRIPT_NAMELEN 128 | ||
34 | #define SCRIPT_MAX_NO 64 | ||
35 | /* | ||
36 | * Usually the full path for a script is: | ||
37 | * /home/username/libexec/perf-core/scripts/python/xxx.py | ||
38 | * /home/username/libexec/perf-core/scripts/perl/xxx.pl | ||
39 | * So 256 should be long enough to contain the full path. | ||
40 | */ | ||
41 | #define SCRIPT_FULLPATH_LEN 256 | ||
42 | |||
43 | /* | ||
44 | * When success, will copy the full path of the selected script | ||
45 | * into the buffer pointed by script_name, and return 0. | ||
46 | * Return -1 on failure. | ||
47 | */ | ||
48 | static int list_scripts(char *script_name) | ||
49 | { | ||
50 | char *buf, *names[SCRIPT_MAX_NO], *paths[SCRIPT_MAX_NO]; | ||
51 | int i, num, choice, ret = -1; | ||
52 | |||
53 | /* Preset the script name to SCRIPT_NAMELEN */ | ||
54 | buf = malloc(SCRIPT_MAX_NO * (SCRIPT_NAMELEN + SCRIPT_FULLPATH_LEN)); | ||
55 | if (!buf) | ||
56 | return ret; | ||
57 | |||
58 | for (i = 0; i < SCRIPT_MAX_NO; i++) { | ||
59 | names[i] = buf + i * (SCRIPT_NAMELEN + SCRIPT_FULLPATH_LEN); | ||
60 | paths[i] = names[i] + SCRIPT_NAMELEN; | ||
61 | } | ||
62 | |||
63 | num = find_scripts(names, paths); | ||
64 | if (num > 0) { | ||
65 | choice = ui__popup_menu(num, names); | ||
66 | if (choice < num && choice >= 0) { | ||
67 | strcpy(script_name, paths[choice]); | ||
68 | ret = 0; | ||
69 | } | ||
70 | } | ||
71 | |||
72 | free(buf); | ||
73 | return ret; | ||
74 | } | ||
75 | |||
76 | static void script_browser__write(struct ui_browser *browser, | ||
77 | void *entry, int row) | ||
78 | { | ||
79 | struct script_line *sline = list_entry(entry, struct script_line, node); | ||
80 | bool current_entry = ui_browser__is_current_entry(browser, row); | ||
81 | |||
82 | ui_browser__set_color(browser, current_entry ? HE_COLORSET_SELECTED : | ||
83 | HE_COLORSET_NORMAL); | ||
84 | |||
85 | slsmg_write_nstring(sline->line, browser->width); | ||
86 | } | ||
87 | |||
88 | static int script_browser__run(struct perf_script_browser *self) | ||
89 | { | ||
90 | int key; | ||
91 | |||
92 | if (ui_browser__show(&self->b, self->script_name, | ||
93 | "Press <- or ESC to exit") < 0) | ||
94 | return -1; | ||
95 | |||
96 | while (1) { | ||
97 | key = ui_browser__run(&self->b, 0); | ||
98 | |||
99 | /* We can add some special key handling here if needed */ | ||
100 | break; | ||
101 | } | ||
102 | |||
103 | ui_browser__hide(&self->b); | ||
104 | return key; | ||
105 | } | ||
106 | |||
107 | |||
108 | int script_browse(const char *script_opt) | ||
109 | { | ||
110 | char cmd[SCRIPT_FULLPATH_LEN*2], script_name[SCRIPT_FULLPATH_LEN]; | ||
111 | char *line = NULL; | ||
112 | size_t len = 0; | ||
113 | ssize_t retlen; | ||
114 | int ret = -1, nr_entries = 0; | ||
115 | FILE *fp; | ||
116 | void *buf; | ||
117 | struct script_line *sline; | ||
118 | |||
119 | struct perf_script_browser script = { | ||
120 | .b = { | ||
121 | .refresh = ui_browser__list_head_refresh, | ||
122 | .seek = ui_browser__list_head_seek, | ||
123 | .write = script_browser__write, | ||
124 | }, | ||
125 | .script_name = script_name, | ||
126 | }; | ||
127 | |||
128 | INIT_LIST_HEAD(&script.entries); | ||
129 | |||
130 | /* Save each line of the output in one struct script_line object. */ | ||
131 | buf = zalloc((sizeof(*sline)) * MAX_LINES); | ||
132 | if (!buf) | ||
133 | return -1; | ||
134 | sline = buf; | ||
135 | |||
136 | memset(script_name, 0, SCRIPT_FULLPATH_LEN); | ||
137 | if (list_scripts(script_name)) | ||
138 | goto exit; | ||
139 | |||
140 | sprintf(cmd, "perf script -s %s ", script_name); | ||
141 | |||
142 | if (script_opt) | ||
143 | strcat(cmd, script_opt); | ||
144 | |||
145 | if (input_name) { | ||
146 | strcat(cmd, " -i "); | ||
147 | strcat(cmd, input_name); | ||
148 | } | ||
149 | |||
150 | strcat(cmd, " 2>&1"); | ||
151 | |||
152 | fp = popen(cmd, "r"); | ||
153 | if (!fp) | ||
154 | goto exit; | ||
155 | |||
156 | while ((retlen = getline(&line, &len, fp)) != -1) { | ||
157 | strncpy(sline->line, line, AVERAGE_LINE_LEN); | ||
158 | |||
159 | /* If one output line is very large, just cut it short */ | ||
160 | if (retlen >= AVERAGE_LINE_LEN) { | ||
161 | sline->line[AVERAGE_LINE_LEN - 1] = '\0'; | ||
162 | sline->line[AVERAGE_LINE_LEN - 2] = '\n'; | ||
163 | } | ||
164 | list_add_tail(&sline->node, &script.entries); | ||
165 | |||
166 | if (script.b.width < retlen) | ||
167 | script.b.width = retlen; | ||
168 | |||
169 | if (nr_entries++ >= MAX_LINES - 1) | ||
170 | break; | ||
171 | sline++; | ||
172 | } | ||
173 | |||
174 | if (script.b.width > AVERAGE_LINE_LEN) | ||
175 | script.b.width = AVERAGE_LINE_LEN; | ||
176 | |||
177 | if (line) | ||
178 | free(line); | ||
179 | pclose(fp); | ||
180 | |||
181 | script.nr_lines = nr_entries; | ||
182 | script.b.nr_entries = nr_entries; | ||
183 | script.b.entries = &script.entries; | ||
184 | |||
185 | ret = script_browser__run(&script); | ||
186 | exit: | ||
187 | free(buf); | ||
188 | return ret; | ||
189 | } | ||