blob: 984de056bfab808da061639203ca4f85afa305b7 [file] [log] [blame]
Harout Hedeshian6202ba72015-04-13 19:02:25 -06001/************************************************************************
2Copyright (c) 2015, The Linux Foundation. All rights reserved.
3
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>
48#include "datatop_interface.h"
49#include "datatop_linked_list.h"
50#include "datatop_opt.h"
51#include "datatop_fileops.h"
52#include "datatop_polling.h"
53#include "datatop_gen_poll.h"
54
55struct dtop_linked_list *first_dpg_list;
56struct cli_opts usr_cl_opts;
57
58/**
59 * @brief Prints the datapoint names and values to the terminal.
60 *
61 * @param dpg_list A pointer to the first node of a linked list which
62 * contains all data_point_gatherer structs to print.
63 */
64void dtop_print_terminal(struct dtop_linked_list *dpg_list)
65{
66 struct dtop_linked_list *curr_ptr = dpg_list;
67 struct dtop_data_point_gatherer *dpset;
68
69 while (curr_ptr) {
70 dpset = (struct dtop_data_point_gatherer *) curr_ptr->data;
71 dtop_print_dpg(dpset);
72 curr_ptr = curr_ptr->next_ptr;
73 }
74}
75
76/**
77 * @brief Polls the data periodically and prints to file specified by the user.
78 *
79 * Polls the data as often as specified by the user in their CLI arguments
80 * and outputs the data to a file also specified in CLI arguments. Then prints
81 * a snapshot of delta(dp_value) to the terminal.
82 *
83 * @param dpg_list A pointer to the first node of a linked list which contains
84 * all data_point_gatherer structs to poll and print.
85 * @param fw A pointer to the file which will be printed to.
86 * @return FILE_ERROR - Writing to file was unsuccessful.
87 * @return FILE_SUCCESS - Writing to file was successful.
88 */
89int dtop_poll_periodically(struct dtop_linked_list *dpg_list, FILE *fw)
90{
91 struct timeval tv, timeout;
92 fd_set rfds;
93 time_t curtime, endtime;
94 int inp, quit;
95 struct dtop_linked_list *curr_ptr = dpg_list;
96 struct dtop_data_point_gatherer *dpset;
97
98 gettimeofday(&tv, NULL);
99 curtime = tv.tv_sec;
100 endtime = tv.tv_sec + usr_cl_opts.poll_time;
101
102 /* print all of our datapoint names as column headers in csv format */
103 if (fprintf(fw, "\"Time\",") < 0)
104 return FILE_ERROR;
105
106 while (curr_ptr) {
107 dpset = (struct dtop_data_point_gatherer *) curr_ptr->data;
108 if (dtop_print_dpg_names_csv(dpset, fw) == FILE_ERROR)
109 return FILE_ERROR;
110 curr_ptr = curr_ptr->next_ptr;
111 }
112 if (fprintf(fw, "\n") < 0)
113 return FILE_ERROR;
114
115 dtop_print_interactive_opts();
116 /* periodically poll the datapoints and print in csv format */
117 while (curtime < endtime
118 || usr_cl_opts.poll_time == POLL_NOT_SPECIFIED) {
119 FD_ZERO(&rfds);
120 FD_SET(0, &rfds);
121 timeout.tv_sec = usr_cl_opts.poll_per;
122 timeout.tv_usec = 0;
123 inp = select(1, &rfds, NULL, NULL, &timeout);
124 if (inp) {
125 char s[4];
126 scanf("%s", s);
127 if (strcmp(s, "quit") == 0
128 || strcmp(s, "q") == 0) {
129 quit = QUIT;
130 break;
131 }
132 if (strcmp(s, "i") == 0) {
133 dtop_print_snapshot_diff(first_dpg_list);
134 dtop_reset_dp_initial_values(first_dpg_list);
135 }
136 if (strcmp(s, "l") == 0)
137 dtop_print_snapshot_diff(first_dpg_list);
138 }
139 gettimeofday(&tv, NULL);
140 curtime = tv.tv_sec;
141 dtop_poll(dpg_list);
142 printf("Polled at %ld.%06ld\n", tv.tv_sec, tv.tv_usec);
143 if (dtop_print_time_at_poll(fw) == FILE_ERROR)
144 return FILE_ERROR;
145 if (dtop_write_pollingdata_csv(dpg_list, fw) == FILE_ERROR)
146 return FILE_ERROR;
147 }
148
149 if (quit != QUIT)
150 dtop_print_snapshot_diff(dpg_list);
151 return FILE_SUCCESS;
152}
153
154static void dtop_set_niceness(int niceness)
155{
156 int pid, rc;
157 pid = getpid();
158 printf("Requesting nice %d\n", niceness);
159 rc = setpriority(PRIO_PROCESS, pid, niceness);
160 if (rc != 0)
161 fprintf(stderr, "Error setting priority [%d]\n", errno);
162
163 rc = getpriority(PRIO_PROCESS, pid);
164 printf("Running with nice %d.\n", rc);
165}
166
167int main(int argc, char **argv)
168{
169 int parse_status;
170 printf("DataTop - Version %s\n", VERSION);
171 printf("(c)2014-2015 Linux Foundation\n");
172
173 dtop_load_default_options(&usr_cl_opts);
174
175 parse_status = dtop_parse_cli_opts(&usr_cl_opts, argc, argv);
176 switch (parse_status) {
177 case PARSE_SUCCESS:
178 dtop_set_niceness(usr_cl_opts.priority);
179 break;
180
181 case PARSE_FORCE_EXIT:
182 exit(EXIT_SUCCESS);
183 break;
184
185 case PARSE_FAILURE:
186 default:
187 printf("Failed to parse command line arguments.\n");
188 exit(EXIT_FAILURE);
189 break;
190 }
191
192 dtop_dual_line_init("/proc/net/netstat");
193 dtop_dual_line_init("/proc/net/snmp");
194 dtop_single_line_init("/proc/net/snmp6");
195 dtop_gen_init("/proc/sys/net/");
196 dtop_gen_init("/sys/module/rmnet_data/parameters/");
197 dtop_gen_init("/sys/class/net/rmnet_mhi0/statistics/");
198 dtop_gen_init("/sys/class/net/usb_rmnet0/statistics/");
199 dtop_gen_init("/sys/class/net/rmnet_ipa0/statistics/");
200 dtop_meminfo_init();
201 dtop_dev_init();
202 dtop_stat_init();
203 dtop_cpu_stats_init();
204 dtop_gen_init("/sys/kernel/debug/clk/bimc_clk/");
205 dtop_gen_init("/sys/kernel/debug/clk/snoc_clk/");
206 dtop_gen_init("/sys/kernel/debug/clk/pnoc_clk/");
207
208 if (usr_cl_opts.print_cl == OPT_CHOSE) {
209 dtop_poll(first_dpg_list);
210 dtop_print_terminal(first_dpg_list);
211 }
212
213 if (usr_cl_opts.print_csv == OPT_CHOSE) {
214 FILE *to_file = NULL;
215 if ((dtop_open_writing_file(usr_cl_opts.file_name,
216 &to_file)) == VALID) {
217 printf("\nData being polled for %ld seconds.\n",
218 usr_cl_opts.poll_time);
219 if (dtop_poll_periodically(first_dpg_list, to_file)
220 == FILE_ERROR) {
221 fprintf(stderr, "err=%d: %s\n", errno,
222 strerror(errno));
223 dtop_close_file(to_file);
224 deconstruct_dpgs(first_dpg_list);
225 dtop_rem_linked_list(first_dpg_list);
226 exit(EXIT_FAILURE);
227 }
228 dtop_close_file(to_file);
229 } else {
230 printf("File Can Not Be Opened\n");
231 exit(EXIT_FAILURE);
232 }
233 }
234
235 if (usr_cl_opts.snapshot_file) {
236 if (dtop_print_system_snapshot(usr_cl_opts.snapshot_file)
237 == FILE_ERROR) {
238 fprintf(stderr, "err=%d: %s\n", errno,
239 strerror(errno));
240 deconstruct_dpgs(first_dpg_list);
241 dtop_rem_linked_list(first_dpg_list);
242 exit(EXIT_FAILURE);
243 }
244 }
245
246 if (usr_cl_opts.print_cl == OPT_NOT_CHOSE &&
247 usr_cl_opts.print_csv == OPT_NOT_CHOSE) {
248 if ((!usr_cl_opts.snapshot_file)
249 || usr_cl_opts.poll_time_selected == POLL_TIME_SELECTED) {
250 printf("\nData will now be polled for %ld seconds.\n",
251 usr_cl_opts.poll_time);
252 dtop_poll(first_dpg_list);
253 sleep(usr_cl_opts.poll_time);
254 dtop_poll(first_dpg_list);
255 dtop_print_snapshot_diff(first_dpg_list);
256 }
257 }
258
259 deconstruct_dpgs(first_dpg_list);
260 dtop_rem_linked_list(first_dpg_list);
261 return 0;
262}
263
264/**
265 * @brief Adds each dpg as a node to a linked list.
266 *
267 * Called when a dpg is initialized.
268 *
269 * @param dpg A pointer to a data_point_gatherer struct which is to be added to the linked list.
270 */
271void dtop_register(struct dtop_data_point_gatherer *dpg)
272{
273 if (dpg)
274 first_dpg_list = dtop_add_linked_list(dpg, first_dpg_list);
275}