blob: fba26e1ab82513fd0922da5f00fbf218962d59b8 [file] [log] [blame]
Harout Hedeshian6202ba72015-04-13 19:02:25 -06001/************************************************************************
Jerome Stanislaus597482d2016-03-04 14:08:46 -07002Copyright (c) 2015-2016, The Linux Foundation. All rights reserved.
Harout Hedeshian6202ba72015-04-13 19:02:25 -06003
4Redistribution and use in source and binary forms, with or without
5modification, are permitted provided that the following conditions are
6met:
7 * Redistributions of source code must retain the above copyright
8 notice, this list of conditions and the following disclaimer.
9 * Redistributions in binary form must reproduce the above
10 copyright notice, this list of conditions and the following
11 disclaimer in the documentation and/or other materials provided
12 with the distribution.
13 * Neither the name of The Linux Foundation nor the names of its
14 contributors may be used to endorse or promote products derived
15 from this software without specific prior written permission.
16
17THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28************************************************************************/
29
30/**
31* @file datatop.c
32* @brief Executes commands for application.
33*
34* Contains the main() function where program executes. Calls appropriate
35* methods based on user's CLI commands. Executes parsing function to
36* determine necessary output and handles errors which may arise during the
37* parse. Initiliazes files for data collection. Will call functions designed
38* to poll and print the data in understandable format.
39*/
40
41#include <stdio.h>
42#include <stdlib.h>
43#include <sys/time.h>
44#include <sys/resource.h>
45#include <unistd.h>
46#include <string.h>
47#include <errno.h>
Harout Hedeshianad3f8482015-06-02 14:43:23 -060048#include <time.h>
Jerome Stanislaus597482d2016-03-04 14:08:46 -070049#include <pthread.h>
Harout Hedeshian6202ba72015-04-13 19:02:25 -060050#include "datatop_interface.h"
51#include "datatop_linked_list.h"
52#include "datatop_opt.h"
53#include "datatop_fileops.h"
54#include "datatop_polling.h"
55#include "datatop_gen_poll.h"
56
57struct dtop_linked_list *first_dpg_list;
58struct cli_opts usr_cl_opts;
59
60/**
61 * @brief Prints the datapoint names and values to the terminal.
62 *
63 * @param dpg_list A pointer to the first node of a linked list which
64 * contains all data_point_gatherer structs to print.
65 */
66void dtop_print_terminal(struct dtop_linked_list *dpg_list)
67{
68 struct dtop_linked_list *curr_ptr = dpg_list;
69 struct dtop_data_point_gatherer *dpset;
70
71 while (curr_ptr) {
72 dpset = (struct dtop_data_point_gatherer *) curr_ptr->data;
73 dtop_print_dpg(dpset);
74 curr_ptr = curr_ptr->next_ptr;
75 }
76}
77
78/**
79 * @brief Polls the data periodically and prints to file specified by the user.
80 *
81 * Polls the data as often as specified by the user in their CLI arguments
82 * and outputs the data to a file also specified in CLI arguments. Then prints
83 * a snapshot of delta(dp_value) to the terminal.
84 *
85 * @param dpg_list A pointer to the first node of a linked list which contains
86 * all data_point_gatherer structs to poll and print.
87 * @param fw A pointer to the file which will be printed to.
88 * @return FILE_ERROR - Writing to file was unsuccessful.
89 * @return FILE_SUCCESS - Writing to file was successful.
90 */
91int dtop_poll_periodically(struct dtop_linked_list *dpg_list, FILE *fw)
92{
93 struct timeval tv, timeout;
94 fd_set rfds;
95 time_t curtime, endtime;
Subash Abhinov Kasiviswanathan6ed645b2016-02-22 20:08:34 -070096 int inp, quit = 0;
Harout Hedeshian6202ba72015-04-13 19:02:25 -060097 struct dtop_linked_list *curr_ptr = dpg_list;
98 struct dtop_data_point_gatherer *dpset;
Harout Hedeshianad3f8482015-06-02 14:43:23 -060099 struct timeval ftime, itime, polltime;
Harout Hedeshian6202ba72015-04-13 19:02:25 -0600100
101 gettimeofday(&tv, NULL);
102 curtime = tv.tv_sec;
103 endtime = tv.tv_sec + usr_cl_opts.poll_time;
104
105 /* print all of our datapoint names as column headers in csv format */
106 if (fprintf(fw, "\"Time\",") < 0)
107 return FILE_ERROR;
108
109 while (curr_ptr) {
110 dpset = (struct dtop_data_point_gatherer *) curr_ptr->data;
111 if (dtop_print_dpg_names_csv(dpset, fw) == FILE_ERROR)
112 return FILE_ERROR;
113 curr_ptr = curr_ptr->next_ptr;
114 }
115 if (fprintf(fw, "\n") < 0)
116 return FILE_ERROR;
117
118 dtop_print_interactive_opts();
Harout Hedeshianad3f8482015-06-02 14:43:23 -0600119 gettimeofday(&itime, NULL);
Harout Hedeshian6202ba72015-04-13 19:02:25 -0600120 /* periodically poll the datapoints and print in csv format */
121 while (curtime < endtime
122 || usr_cl_opts.poll_time == POLL_NOT_SPECIFIED) {
123 FD_ZERO(&rfds);
124 FD_SET(0, &rfds);
125 timeout.tv_sec = usr_cl_opts.poll_per;
126 timeout.tv_usec = 0;
Harout Hedeshianad3f8482015-06-02 14:43:23 -0600127 //ftime is right before timeout calculations for most acurate calculations
128 gettimeofday(&ftime, NULL);
129 timersub(&ftime, &itime, &polltime);
130 timersub(&timeout,&polltime, &timeout);
Harout Hedeshian6202ba72015-04-13 19:02:25 -0600131 inp = select(1, &rfds, NULL, NULL, &timeout);
Harout Hedeshianad3f8482015-06-02 14:43:23 -0600132 gettimeofday(&itime, NULL);
Harout Hedeshian6202ba72015-04-13 19:02:25 -0600133 if (inp) {
134 char s[4];
135 scanf("%s", s);
136 if (strcmp(s, "quit") == 0
137 || strcmp(s, "q") == 0) {
138 quit = QUIT;
139 break;
140 }
141 if (strcmp(s, "i") == 0) {
142 dtop_print_snapshot_diff(first_dpg_list);
143 dtop_reset_dp_initial_values(first_dpg_list);
144 }
145 if (strcmp(s, "l") == 0)
146 dtop_print_snapshot_diff(first_dpg_list);
147 }
148 gettimeofday(&tv, NULL);
149 curtime = tv.tv_sec;
150 dtop_poll(dpg_list);
151 printf("Polled at %ld.%06ld\n", tv.tv_sec, tv.tv_usec);
152 if (dtop_print_time_at_poll(fw) == FILE_ERROR)
153 return FILE_ERROR;
154 if (dtop_write_pollingdata_csv(dpg_list, fw) == FILE_ERROR)
155 return FILE_ERROR;
156 }
157
158 if (quit != QUIT)
159 dtop_print_snapshot_diff(dpg_list);
160 return FILE_SUCCESS;
161}
162
163static void dtop_set_niceness(int niceness)
164{
165 int pid, rc;
166 pid = getpid();
167 printf("Requesting nice %d\n", niceness);
168 rc = setpriority(PRIO_PROCESS, pid, niceness);
169 if (rc != 0)
170 fprintf(stderr, "Error setting priority [%d]\n", errno);
171
172 rc = getpriority(PRIO_PROCESS, pid);
173 printf("Running with nice %d.\n", rc);
174}
175
176int main(int argc, char **argv)
177{
178 int parse_status;
Jerome Stanislaus597482d2016-03-04 14:08:46 -0700179 pthread_t tid;
Harout Hedeshian6202ba72015-04-13 19:02:25 -0600180 printf("DataTop - Version %s\n", VERSION);
181 printf("(c)2014-2015 Linux Foundation\n");
182
183 dtop_load_default_options(&usr_cl_opts);
184
185 parse_status = dtop_parse_cli_opts(&usr_cl_opts, argc, argv);
186 switch (parse_status) {
187 case PARSE_SUCCESS:
188 dtop_set_niceness(usr_cl_opts.priority);
189 break;
190
191 case PARSE_FORCE_EXIT:
192 exit(EXIT_SUCCESS);
193 break;
194
195 case PARSE_FAILURE:
196 default:
197 printf("Failed to parse command line arguments.\n");
198 exit(EXIT_FAILURE);
199 break;
200 }
201
Jerome Stanislaus597482d2016-03-04 14:08:46 -0700202 if (usr_cl_opts.iptables_rules_routes == OPT_CHOSE) {
203 if (!usr_cl_opts.out_dir) {
204 printf("Please provide an out directory.\n");
205 exit(EXIT_FAILURE);
206 }
207 }
208
Harout Hedeshian6202ba72015-04-13 19:02:25 -0600209 dtop_dual_line_init("/proc/net/netstat");
210 dtop_dual_line_init("/proc/net/snmp");
211 dtop_single_line_init("/proc/net/snmp6");
212 dtop_gen_init("/proc/sys/net/");
213 dtop_gen_init("/sys/module/rmnet_data/parameters/");
214 dtop_gen_init("/sys/class/net/rmnet_mhi0/statistics/");
215 dtop_gen_init("/sys/class/net/usb_rmnet0/statistics/");
216 dtop_gen_init("/sys/class/net/rmnet_ipa0/statistics/");
217 dtop_meminfo_init();
218 dtop_dev_init();
219 dtop_stat_init();
220 dtop_cpu_stats_init();
221 dtop_gen_init("/sys/kernel/debug/clk/bimc_clk/");
222 dtop_gen_init("/sys/kernel/debug/clk/snoc_clk/");
223 dtop_gen_init("/sys/kernel/debug/clk/pnoc_clk/");
224
Jerome Stanislaus597482d2016-03-04 14:08:46 -0700225 if (usr_cl_opts.iptables_rules_routes == OPT_CHOSE) {
226 printf("Datatop IP Tables, rules, routes\n");
227 dtop_ip_table_init(usr_cl_opts.out_dir);
228 if(0 != pthread_create(&tid, NULL, &dtop_ip_table_start_poll, NULL)) {
229 printf("Unable to create capture_ip_tables_rules_routes thread\n");
230 }
231 }
232
Harout Hedeshian6202ba72015-04-13 19:02:25 -0600233 if (usr_cl_opts.print_cl == OPT_CHOSE) {
234 dtop_poll(first_dpg_list);
235 dtop_print_terminal(first_dpg_list);
236 }
237
238 if (usr_cl_opts.print_csv == OPT_CHOSE) {
239 FILE *to_file = NULL;
240 if ((dtop_open_writing_file(usr_cl_opts.file_name,
241 &to_file)) == VALID) {
242 printf("\nData being polled for %ld seconds.\n",
243 usr_cl_opts.poll_time);
244 if (dtop_poll_periodically(first_dpg_list, to_file)
245 == FILE_ERROR) {
246 fprintf(stderr, "err=%d: %s\n", errno,
247 strerror(errno));
248 dtop_close_file(to_file);
249 deconstruct_dpgs(first_dpg_list);
250 dtop_rem_linked_list(first_dpg_list);
251 exit(EXIT_FAILURE);
252 }
253 dtop_close_file(to_file);
254 } else {
255 printf("File Can Not Be Opened\n");
256 exit(EXIT_FAILURE);
257 }
258 }
259
260 if (usr_cl_opts.snapshot_file) {
261 if (dtop_print_system_snapshot(usr_cl_opts.snapshot_file)
262 == FILE_ERROR) {
263 fprintf(stderr, "err=%d: %s\n", errno,
264 strerror(errno));
265 deconstruct_dpgs(first_dpg_list);
266 dtop_rem_linked_list(first_dpg_list);
267 exit(EXIT_FAILURE);
268 }
269 }
270
271 if (usr_cl_opts.print_cl == OPT_NOT_CHOSE &&
272 usr_cl_opts.print_csv == OPT_NOT_CHOSE) {
273 if ((!usr_cl_opts.snapshot_file)
274 || usr_cl_opts.poll_time_selected == POLL_TIME_SELECTED) {
275 printf("\nData will now be polled for %ld seconds.\n",
276 usr_cl_opts.poll_time);
277 dtop_poll(first_dpg_list);
278 sleep(usr_cl_opts.poll_time);
279 dtop_poll(first_dpg_list);
280 dtop_print_snapshot_diff(first_dpg_list);
281 }
282 }
283
284 deconstruct_dpgs(first_dpg_list);
285 dtop_rem_linked_list(first_dpg_list);
286 return 0;
287}
288
289/**
290 * @brief Adds each dpg as a node to a linked list.
291 *
292 * Called when a dpg is initialized.
293 *
294 * @param dpg A pointer to a data_point_gatherer struct which is to be added to the linked list.
295 */
296void dtop_register(struct dtop_data_point_gatherer *dpg)
297{
298 if (dpg)
299 first_dpg_list = dtop_add_linked_list(dpg, first_dpg_list);
300}