| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2008 The Android Open Source Project | 
|  | 3 | * | 
|  | 4 | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | 5 | * you may not use this file except in compliance with the License. | 
|  | 6 | * You may obtain a copy of the License at | 
|  | 7 | * | 
|  | 8 | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | 9 | * | 
|  | 10 | * Unless required by applicable law or agreed to in writing, software | 
|  | 11 | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 13 | * See the License for the specific language governing permissions and | 
|  | 14 | * limitations under the License. | 
|  | 15 | */ | 
|  | 16 |  | 
|  | 17 | #ifndef _DUMPSTATE_H_ | 
|  | 18 | #define _DUMPSTATE_H_ | 
|  | 19 |  | 
| Felipe Leme | 93d705b | 2015-11-10 20:10:25 -0800 | [diff] [blame] | 20 | /* When defined, skips the real dumps and just print the section headers. | 
|  | 21 | Useful when debugging dumpstate itself. */ | 
|  | 22 | //#define _DUMPSTATE_DRY_RUN_ | 
|  | 23 |  | 
|  | 24 | #ifdef _DUMPSTATE_DRY_RUN_ | 
|  | 25 | #define ON_DRY_RUN_RETURN(X) return X | 
| Christopher Ferris | ed24d2a | 2015-11-12 14:01:56 -0800 | [diff] [blame] | 26 | #define ON_DRY_RUN(code) code | 
|  | 27 | #else | 
| Felipe Leme | 93d705b | 2015-11-10 20:10:25 -0800 | [diff] [blame] | 28 | #define ON_DRY_RUN_RETURN(X) | 
| Christopher Ferris | ed24d2a | 2015-11-12 14:01:56 -0800 | [diff] [blame] | 29 | #define ON_DRY_RUN(code) | 
| Felipe Leme | 93d705b | 2015-11-10 20:10:25 -0800 | [diff] [blame] | 30 | #endif | 
|  | 31 |  | 
| Felipe Leme | cbce55d | 2016-02-08 09:53:18 -0800 | [diff] [blame] | 32 | #ifndef MYLOGD | 
| Felipe Leme | 60869c9 | 2016-02-09 16:07:20 -0800 | [diff] [blame] | 33 | #define MYLOGD(...) fprintf(stderr, __VA_ARGS__); ALOGD(__VA_ARGS__); | 
| Felipe Leme | cbce55d | 2016-02-08 09:53:18 -0800 | [diff] [blame] | 34 | #endif | 
|  | 35 |  | 
|  | 36 | #ifndef MYLOGI | 
| Felipe Leme | 60869c9 | 2016-02-09 16:07:20 -0800 | [diff] [blame] | 37 | #define MYLOGI(...) fprintf(stderr, __VA_ARGS__); ALOGI(__VA_ARGS__); | 
| Felipe Leme | cbce55d | 2016-02-08 09:53:18 -0800 | [diff] [blame] | 38 | #endif | 
|  | 39 |  | 
|  | 40 | #ifndef MYLOGE | 
| Felipe Leme | 60869c9 | 2016-02-09 16:07:20 -0800 | [diff] [blame] | 41 | #define MYLOGE(...) fprintf(stderr, __VA_ARGS__); ALOGE(__VA_ARGS__); | 
| Felipe Leme | cbce55d | 2016-02-08 09:53:18 -0800 | [diff] [blame] | 42 | #endif | 
| Felipe Leme | 93d705b | 2015-11-10 20:10:25 -0800 | [diff] [blame] | 43 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 44 | #include <time.h> | 
|  | 45 | #include <unistd.h> | 
| Colin Cross | 0c22e8b | 2012-11-02 15:46:56 -0700 | [diff] [blame] | 46 | #include <stdbool.h> | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 47 | #include <stdio.h> | 
| Felipe Leme | 36b3f6f | 2015-11-19 15:41:04 -0800 | [diff] [blame] | 48 | #include <vector> | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 49 |  | 
|  | 50 | #define SU_PATH "/system/xbin/su" | 
|  | 51 |  | 
| Felipe Leme | 8620bb4 | 2015-11-10 11:04:45 -0800 | [diff] [blame] | 52 | #ifdef __cplusplus | 
|  | 53 | extern "C" { | 
|  | 54 | #endif | 
|  | 55 |  | 
| Colin Cross | 0c22e8b | 2012-11-02 15:46:56 -0700 | [diff] [blame] | 56 | typedef void (for_each_pid_func)(int, const char *); | 
|  | 57 | typedef void (for_each_tid_func)(int, int, const char *); | 
|  | 58 |  | 
| Felipe Leme | 71bbfc5 | 2015-11-23 14:14:51 -0800 | [diff] [blame] | 59 | /* Estimated total weight of bugreport generation. | 
|  | 60 | * | 
|  | 61 | * Each section contributes to the total weight by an individual weight, so the overall progress | 
|  | 62 | * can be calculated by dividing the all completed weight by the total weight. | 
|  | 63 | * | 
|  | 64 | * This value is defined empirically and it need to be adjusted as more sections are added. | 
| Felipe Leme | ad5f6c4 | 2015-11-30 14:26:46 -0800 | [diff] [blame] | 65 | * | 
|  | 66 | * It does not need to match the exact sum of all sections, but ideally it should to be slight more | 
|  | 67 | * than such sum: a value too high will cause the bugreport to finish before the user expected (for | 
| Felipe Leme | faf67e3 | 2016-03-28 11:15:22 -0700 | [diff] [blame] | 68 | * example, jumping from 70% to 100%), while a value too low will cause the progress to get stuck | 
|  | 69 | * at an almost-finished value (like 99%) for a while. | 
| Felipe Leme | 71bbfc5 | 2015-11-23 14:14:51 -0800 | [diff] [blame] | 70 | */ | 
| Felipe Leme | faf67e3 | 2016-03-28 11:15:22 -0700 | [diff] [blame] | 71 | static const int WEIGHT_TOTAL = 6500; | 
| Felipe Leme | 71bbfc5 | 2015-11-23 14:14:51 -0800 | [diff] [blame] | 72 |  | 
|  | 73 | /* Most simple commands have 10 as timeout, so 5 is a good estimate */ | 
|  | 74 | static const int WEIGHT_FILE = 5; | 
|  | 75 |  | 
|  | 76 | /* | 
| Felipe Leme | 0c80cf0 | 2016-01-05 13:25:34 -0800 | [diff] [blame] | 77 | * TODO: the dumpstate internal state is getting fragile; for example, this variable is defined | 
| Felipe Leme | 71bbfc5 | 2015-11-23 14:14:51 -0800 | [diff] [blame] | 78 | * here, declared at utils.cpp, and used on utils.cpp and dumpstate.cpp. | 
|  | 79 | * It would be better to take advantage of the C++ migration and encapsulate the state in an object, | 
|  | 80 | * but that will be better handled in a major C++ refactoring, which would also get rid of other C | 
|  | 81 | * idioms (like using std::string instead of char*, removing varargs, etc...) */ | 
| Felipe Leme | 02b7e00 | 2016-07-22 12:03:20 -0700 | [diff] [blame] | 82 | extern int do_update_progress, progress, weight_total, control_socket_fd; | 
| Felipe Leme | 71bbfc5 | 2015-11-23 14:14:51 -0800 | [diff] [blame] | 83 |  | 
| Felipe Leme | 71ca15e | 2016-05-19 16:18:17 -0700 | [diff] [blame] | 84 | /* full path of the directory where the bugreport files will be written */ | 
|  | 85 | extern std::string bugreport_dir; | 
|  | 86 |  | 
|  | 87 | /* root dir for all files copied as-is into the bugreport. */ | 
|  | 88 | extern const std::string ZIP_ROOT_DIR; | 
|  | 89 |  | 
| Felipe Leme | 0397498 | 2016-10-21 11:38:26 -0700 | [diff] [blame] | 90 | /* Checkes whether dumpstate is generating a zipped bugreport. */ | 
|  | 91 | bool is_zipping(); | 
|  | 92 |  | 
| Felipe Leme | 71ca15e | 2016-05-19 16:18:17 -0700 | [diff] [blame] | 93 | /* adds a new entry to the existing zip file. */ | 
|  | 94 | bool add_zip_entry(const std::string& entry_name, const std::string& entry_path); | 
|  | 95 |  | 
|  | 96 | /* adds a new entry to the existing zip file. */ | 
|  | 97 | bool add_zip_entry_from_fd(const std::string& entry_name, int fd); | 
|  | 98 |  | 
| Calvin On | 249beee | 2016-06-03 15:17:07 -0700 | [diff] [blame] | 99 | /* adds all files from a directory to the zipped bugreport file */ | 
|  | 100 | void add_dir(const char *dir, bool recursive); | 
|  | 101 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 102 | /* prints the contents of a file */ | 
| Christopher Ferris | 1fe6107 | 2014-07-22 16:08:19 -0700 | [diff] [blame] | 103 | int dump_file(const char *title, const char *path); | 
|  | 104 |  | 
| Felipe Leme | 71a74ac | 2016-03-17 15:43:25 -0700 | [diff] [blame] | 105 | /* saves the the contents of a file as a long */ | 
|  | 106 | int read_file_as_long(const char *path, long int *output); | 
|  | 107 |  | 
| Christopher Ferris | 54bcc5f | 2015-02-10 12:15:01 -0800 | [diff] [blame] | 108 | /* prints the contents of the fd | 
|  | 109 | * fd must have been opened with the flag O_NONBLOCK. | 
|  | 110 | */ | 
| Christopher Ferris | 1fe6107 | 2014-07-22 16:08:19 -0700 | [diff] [blame] | 111 | int dump_file_from_fd(const char *title, const char *path, int fd); | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 112 |  | 
| Mark Salyzyn | 326842f | 2015-04-30 09:49:41 -0700 | [diff] [blame] | 113 | /* calls skip to gate calling dump_from_fd recursively | 
|  | 114 | * in the specified directory. dump_from_fd defaults to | 
|  | 115 | * dump_file_from_fd above when set to NULL. skip defaults | 
|  | 116 | * to false when set to NULL. dump_from_fd will always be | 
|  | 117 | * called with title NULL. | 
|  | 118 | */ | 
|  | 119 | int dump_files(const char *title, const char *dir, | 
|  | 120 | bool (*skip)(const char *path), | 
|  | 121 | int (*dump_from_fd)(const char *title, const char *path, int fd)); | 
|  | 122 |  | 
| Felipe Leme | cf6a8b4 | 2016-03-11 10:38:19 -0800 | [diff] [blame] | 123 | // TODO: need to refactor all those run_command variations; there shold be just one, receiving an | 
|  | 124 | // optional CommandOptions objects with values such as run_always, drop_root, etc... | 
|  | 125 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 126 | /* forks a command and waits for it to finish -- terminate args with NULL */ | 
| Felipe Leme | cf6a8b4 | 2016-03-11 10:38:19 -0800 | [diff] [blame] | 127 | int run_command_as_shell(const char *title, int timeout_seconds, const char *command, ...); | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 128 | int run_command(const char *title, int timeout_seconds, const char *command, ...); | 
|  | 129 |  | 
| Felipe Leme | 29c3971 | 2016-04-01 10:02:00 -0700 | [diff] [blame] | 130 | enum RootMode { DROP_ROOT, DONT_DROP_ROOT }; | 
|  | 131 | enum StdoutMode { NORMAL_STDOUT, REDIRECT_TO_STDERR }; | 
|  | 132 |  | 
| Felipe Leme | 93d705b | 2015-11-10 20:10:25 -0800 | [diff] [blame] | 133 | /* forks a command and waits for it to finish | 
|  | 134 | first element of args is the command, and last must be NULL. | 
|  | 135 | command is always ran, even when _DUMPSTATE_DRY_RUN_ is defined. */ | 
| Felipe Leme | 29c3971 | 2016-04-01 10:02:00 -0700 | [diff] [blame] | 136 | int run_command_always(const char *title, RootMode root_mode, StdoutMode stdout_mode, | 
|  | 137 | int timeout_seconds, const char *args[]); | 
| Felipe Leme | cf6a8b4 | 2016-03-11 10:38:19 -0800 | [diff] [blame] | 138 |  | 
|  | 139 | /* switch to non-root user and group */ | 
|  | 140 | bool drop_root_user(); | 
| Felipe Leme | 93d705b | 2015-11-10 20:10:25 -0800 | [diff] [blame] | 141 |  | 
| Felipe Leme | 36b3f6f | 2015-11-19 15:41:04 -0800 | [diff] [blame] | 142 | /* sends a broadcast using Activity Manager */ | 
|  | 143 | void send_broadcast(const std::string& action, const std::vector<std::string>& args); | 
|  | 144 |  | 
| Felipe Leme | 71bbfc5 | 2015-11-23 14:14:51 -0800 | [diff] [blame] | 145 | /* updates the overall progress of dumpstate by the given weight increment */ | 
|  | 146 | void update_progress(int weight); | 
|  | 147 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 148 | /* prints all the system properties */ | 
|  | 149 | void print_properties(); | 
|  | 150 |  | 
| Felipe Leme | 2628e9e | 2016-04-12 16:36:51 -0700 | [diff] [blame] | 151 | /** opens a socket and returns its file descriptor */ | 
|  | 152 | int open_socket(const char *service); | 
|  | 153 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 154 | /* redirect output to a service control socket */ | 
|  | 155 | void redirect_to_socket(FILE *redirect, const char *service); | 
|  | 156 |  | 
| Christopher Ferris | ff4a4dc | 2015-02-09 16:24:47 -0800 | [diff] [blame] | 157 | /* redirect output to a file */ | 
|  | 158 | void redirect_to_file(FILE *redirect, char *path); | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 159 |  | 
| Felipe Leme | 111b9d0 | 2016-02-03 09:28:24 -0800 | [diff] [blame] | 160 | /* create leading directories, if necessary */ | 
|  | 161 | void create_parent_dirs(const char *path); | 
|  | 162 |  | 
| Jeff Brown | bf7f492 | 2012-06-07 16:40:01 -0700 | [diff] [blame] | 163 | /* dump Dalvik and native stack traces, return the trace file location (NULL if none) */ | 
|  | 164 | const char *dump_traces(); | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 165 |  | 
|  | 166 | /* for each process in the system, run the specified function */ | 
| Colin Cross | 0c22e8b | 2012-11-02 15:46:56 -0700 | [diff] [blame] | 167 | void for_each_pid(for_each_pid_func func, const char *header); | 
|  | 168 |  | 
|  | 169 | /* for each thread in the system, run the specified function */ | 
|  | 170 | void for_each_tid(for_each_tid_func func, const char *header); | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 171 |  | 
|  | 172 | /* Displays a blocked processes in-kernel wait channel */ | 
| Colin Cross | 0c22e8b | 2012-11-02 15:46:56 -0700 | [diff] [blame] | 173 | void show_wchan(int pid, int tid, const char *name); | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 174 |  | 
| Mark Salyzyn | a297c32 | 2016-02-05 15:33:17 -0800 | [diff] [blame] | 175 | /* Displays a processes times */ | 
|  | 176 | void show_showtime(int pid, const char *name); | 
|  | 177 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 178 | /* Runs "showmap" for a process */ | 
|  | 179 | void do_showmap(int pid, const char *name); | 
|  | 180 |  | 
|  | 181 | /* Gets the dmesg output for the kernel */ | 
|  | 182 | void do_dmesg(); | 
|  | 183 |  | 
| Sreeram Ramachandran | 2b3bba3 | 2014-07-08 15:40:55 -0700 | [diff] [blame] | 184 | /* Prints the contents of all the routing tables, both IPv4 and IPv6. */ | 
|  | 185 | void dump_route_tables(); | 
|  | 186 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 187 | /* Play a sound via Stagefright */ | 
| Christopher Ferris | 1fe6107 | 2014-07-22 16:08:19 -0700 | [diff] [blame] | 188 | void play_sound(const char *path); | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 189 |  | 
|  | 190 | /* Implemented by libdumpstate_board to dump board-specific info */ | 
|  | 191 | void dumpstate_board(); | 
|  | 192 |  | 
| Felipe Leme | e338bf6 | 2015-12-07 14:03:50 -0800 | [diff] [blame] | 193 | /* Takes a screenshot and save it to the given file */ | 
| Felipe Leme | 3634a1e | 2015-12-09 10:11:47 -0800 | [diff] [blame] | 194 | void take_screenshot(const std::string& path); | 
| Felipe Leme | e338bf6 | 2015-12-07 14:03:50 -0800 | [diff] [blame] | 195 |  | 
| Felipe Leme | 0c80cf0 | 2016-01-05 13:25:34 -0800 | [diff] [blame] | 196 | /* Vibrates for a given durating (in milliseconds). */ | 
|  | 197 | void vibrate(FILE* vibrator, int ms); | 
|  | 198 |  | 
|  | 199 | /* Checks if a given path is a directory. */ | 
|  | 200 | bool is_dir(const char* pathname); | 
|  | 201 |  | 
|  | 202 | /** Gets the last modification time of a file, or default time if file is not found. */ | 
|  | 203 | time_t get_mtime(int fd, time_t default_mtime); | 
|  | 204 |  | 
| Calvin On | 249beee | 2016-06-03 15:17:07 -0700 | [diff] [blame] | 205 | /* Dumps eMMC Extended CSD data. */ | 
| Felipe Leme | 78f2c86 | 2015-12-21 09:55:22 -0800 | [diff] [blame] | 206 | void dump_emmc_ecsd(const char *ext_csd_path); | 
|  | 207 |  | 
| Calvin On | 249beee | 2016-06-03 15:17:07 -0700 | [diff] [blame] | 208 | /** Gets command-line arguments. */ | 
| Felipe Leme | a34efb7 | 2016-03-11 09:33:32 -0800 | [diff] [blame] | 209 | void format_args(int argc, const char *argv[], std::string *args); | 
| Felipe Leme | 88c7933 | 2016-02-22 11:06:49 -0800 | [diff] [blame] | 210 |  | 
| Calvin On | 249beee | 2016-06-03 15:17:07 -0700 | [diff] [blame] | 211 | /** Tells if the device is running a user build. */ | 
|  | 212 | bool is_user_build(); | 
|  | 213 |  | 
| Felipe Leme | 78f2c86 | 2015-12-21 09:55:22 -0800 | [diff] [blame] | 214 | /* | 
|  | 215 | * Helper class used to report how long it takes for a section to finish. | 
|  | 216 | * | 
|  | 217 | * Typical usage: | 
|  | 218 | * | 
|  | 219 | *    DurationReporter duration_reporter(title); | 
|  | 220 | * | 
|  | 221 | */ | 
|  | 222 | class DurationReporter { | 
|  | 223 | public: | 
| Chih-Hung Hsieh | 759af11 | 2016-09-01 11:42:49 -0700 | [diff] [blame] | 224 | explicit DurationReporter(const char *title); | 
| Felipe Leme | 608385d | 2016-02-01 10:35:38 -0800 | [diff] [blame] | 225 | DurationReporter(const char *title, FILE* out); | 
| Felipe Leme | 78f2c86 | 2015-12-21 09:55:22 -0800 | [diff] [blame] | 226 |  | 
|  | 227 | ~DurationReporter(); | 
|  | 228 |  | 
|  | 229 | static uint64_t nanotime(); | 
|  | 230 |  | 
|  | 231 | private: | 
|  | 232 | const char* title_; | 
| Felipe Leme | 608385d | 2016-02-01 10:35:38 -0800 | [diff] [blame] | 233 | FILE* out_; | 
| Felipe Leme | 78f2c86 | 2015-12-21 09:55:22 -0800 | [diff] [blame] | 234 | uint64_t started_; | 
|  | 235 | }; | 
|  | 236 |  | 
| Felipe Leme | 8620bb4 | 2015-11-10 11:04:45 -0800 | [diff] [blame] | 237 | #ifdef __cplusplus | 
|  | 238 | } | 
|  | 239 | #endif | 
|  | 240 |  | 
| Colin Cross | f45fa6b | 2012-03-26 12:38:26 -0700 | [diff] [blame] | 241 | #endif /* _DUMPSTATE_H_ */ |