Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 1 | /************************************************************************ |
Jerome Stanislaus | 597482d | 2016-03-04 14:08:46 -0700 | [diff] [blame] | 2 | Copyright (c) 2015-2016, The Linux Foundation. All rights reserved. |
Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 3 | |
| 4 | Redistribution and use in source and binary forms, with or without |
| 5 | modification, are permitted provided that the following conditions are |
| 6 | met: |
| 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 | |
| 17 | THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED |
| 18 | WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
| 19 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT |
| 20 | ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS |
| 21 | BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| 22 | CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| 23 | SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR |
| 24 | BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, |
| 25 | WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE |
| 26 | OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN |
| 27 | IF 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 Hedeshian | ad3f848 | 2015-06-02 14:43:23 -0600 | [diff] [blame] | 48 | #include <time.h> |
Jerome Stanislaus | 597482d | 2016-03-04 14:08:46 -0700 | [diff] [blame] | 49 | #include <pthread.h> |
Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 50 | #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 | |
| 57 | struct dtop_linked_list *first_dpg_list; |
| 58 | struct 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 | */ |
| 66 | void 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 | */ |
| 91 | int 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 Kasiviswanathan | 6ed645b | 2016-02-22 20:08:34 -0700 | [diff] [blame] | 96 | int inp, quit = 0; |
Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 97 | struct dtop_linked_list *curr_ptr = dpg_list; |
| 98 | struct dtop_data_point_gatherer *dpset; |
Harout Hedeshian | ad3f848 | 2015-06-02 14:43:23 -0600 | [diff] [blame] | 99 | struct timeval ftime, itime, polltime; |
Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 100 | |
| 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 Hedeshian | ad3f848 | 2015-06-02 14:43:23 -0600 | [diff] [blame] | 119 | gettimeofday(&itime, NULL); |
Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 120 | /* 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 Hedeshian | ad3f848 | 2015-06-02 14:43:23 -0600 | [diff] [blame] | 127 | //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 Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 131 | inp = select(1, &rfds, NULL, NULL, &timeout); |
Harout Hedeshian | ad3f848 | 2015-06-02 14:43:23 -0600 | [diff] [blame] | 132 | gettimeofday(&itime, NULL); |
Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 133 | 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 | |
| 163 | static 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 | |
| 176 | int main(int argc, char **argv) |
| 177 | { |
| 178 | int parse_status; |
Jerome Stanislaus | 597482d | 2016-03-04 14:08:46 -0700 | [diff] [blame] | 179 | pthread_t tid; |
Harout Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 180 | 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 Stanislaus | 597482d | 2016-03-04 14:08:46 -0700 | [diff] [blame] | 202 | 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 Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 209 | 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 Stanislaus | 597482d | 2016-03-04 14:08:46 -0700 | [diff] [blame] | 225 | 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 Hedeshian | 6202ba7 | 2015-04-13 19:02:25 -0600 | [diff] [blame] | 233 | 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 | */ |
| 296 | void 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 | } |