// SPDX-License-Identifier: LGPL-2.1-or-later /* * * BlueZ - Bluetooth protocol stack for Linux * * Copyright (C) 2011-2014 Intel Corporation * Copyright (C) 2002-2010 Marcel Holtmann * * */ #ifdef HAVE_CONFIG_H #include #endif #define _GNU_SOURCE #include #include #include #include #include #include #include "src/shared/mainloop.h" #include "src/shared/tty.h" #include "packet.h" #include "lmp.h" #include "keys.h" #include "analyze.h" #include "ellisys.h" #include "control.h" #include "display.h" static void signal_callback(int signum, void *user_data) { switch (signum) { case SIGINT: case SIGTERM: mainloop_quit(); break; } } static void usage(void) { printf("btmon - Bluetooth monitor\n" "Usage:\n"); printf("\tbtmon [options]\n"); printf("options:\n" "\t-r, --read Read traces in btsnoop format\n" "\t-w, --write Save traces in btsnoop format\n" "\t-a, --analyze Analyze traces in btsnoop format\n" "\t-s, --server Start monitor server socket\n" "\t-p, --priority Show only priority or lower\n" "\t-i, --index Show only specified controller\n" "\t-d, --tty Read data from TTY\n" "\t-B, --tty-speed Set TTY speed (default 115200)\n" "\t-V, --vendor Set default company identifier\n" "\t-M, --mgmt Open channel for mgmt events\n" "\t-t, --time Show time instead of time offset\n" "\t-T, --date Show time and date information\n" "\t-S, --sco Dump SCO traffic\n" "\t-A, --a2dp Dump A2DP stream traffic\n" "\t-E, --ellisys [ip] Send Ellisys HCI Injection\n" "\t-P, --no-pager Disable pager usage\n" "\t-J --jlink ,[],[],[]\n" "\t Read data from RTT\n" "\t-R --rtt [
],[],[]\n" "\t RTT control block parameters\n" "\t-C, --columns [width] Output width if not a terminal\n" "\t-c, --color [mode] Output color: auto/always/never\n" "\t-h, --help Show help options\n"); } static const struct option main_options[] = { { "read", required_argument, NULL, 'r' }, { "write", required_argument, NULL, 'w' }, { "analyze", required_argument, NULL, 'a' }, { "server", required_argument, NULL, 's' }, { "priority", required_argument, NULL, 'p' }, { "index", required_argument, NULL, 'i' }, { "tty", required_argument, NULL, 'd' }, { "tty-speed", required_argument, NULL, 'B' }, { "vendor", required_argument, NULL, 'V' }, { "mgmt", no_argument, NULL, 'M' }, { "no-time", no_argument, NULL, 'N' }, { "time", no_argument, NULL, 't' }, { "date", no_argument, NULL, 'T' }, { "sco", no_argument, NULL, 'S' }, { "a2dp", no_argument, NULL, 'A' }, { "ellisys", required_argument, NULL, 'E' }, { "no-pager", no_argument, NULL, 'P' }, { "jlink", required_argument, NULL, 'J' }, { "rtt", required_argument, NULL, 'R' }, { "columns", required_argument, NULL, 'C' }, { "color", required_argument, NULL, 'c' }, { "todo", no_argument, NULL, '#' }, { "version", no_argument, NULL, 'v' }, { "help", no_argument, NULL, 'h' }, { } }; int main(int argc, char *argv[]) { unsigned long filter_mask = 0; bool use_pager = true; const char *reader_path = NULL; const char *writer_path = NULL; const char *analyze_path = NULL; const char *ellisys_server = NULL; const char *tty = NULL; unsigned int tty_speed = B115200; unsigned short ellisys_port = 0; const char *str; char *jlink = NULL; char *rtt = NULL; int exit_status; mainloop_init(); filter_mask |= PACKET_FILTER_SHOW_TIME_OFFSET; for (;;) { int opt; struct sockaddr_un addr; opt = getopt_long(argc, argv, "r:w:a:s:p:i:d:B:V:MNtTSAE:PJ:R:C:c:vh", main_options, NULL); if (opt < 0) break; switch (opt) { case 'r': reader_path = optarg; break; case 'w': writer_path = optarg; break; case 'a': analyze_path = optarg; break; case 's': if (strlen(optarg) > sizeof(addr.sun_path) - 1) { fprintf(stderr, "Socket name too long\n"); return EXIT_FAILURE; } control_server(optarg); break; case 'p': packet_set_priority(optarg); break; case 'i': if (strlen(optarg) > 3 && !strncmp(optarg, "hci", 3)) str = optarg + 3; else str = optarg; if (!isdigit(*str)) { usage(); return EXIT_FAILURE; } packet_select_index(atoi(str)); break; case 'd': tty = optarg; break; case 'B': tty_speed = tty_get_speed(atoi(optarg)); if (!tty_speed) { fprintf(stderr, "Unknown speed: %s\n", optarg); return EXIT_FAILURE; } break; case 'V': str = optarg; packet_set_fallback_manufacturer(atoi(str)); break; case 'M': filter_mask |= PACKET_FILTER_SHOW_MGMT_SOCKET; break; case 'N': filter_mask &= ~PACKET_FILTER_SHOW_TIME_OFFSET; break; case 't': filter_mask &= ~PACKET_FILTER_SHOW_TIME_OFFSET; filter_mask |= PACKET_FILTER_SHOW_TIME; break; case 'T': filter_mask &= ~PACKET_FILTER_SHOW_TIME_OFFSET; filter_mask |= PACKET_FILTER_SHOW_TIME; filter_mask |= PACKET_FILTER_SHOW_DATE; break; case 'S': filter_mask |= PACKET_FILTER_SHOW_SCO_DATA; break; case 'A': filter_mask |= PACKET_FILTER_SHOW_A2DP_STREAM; break; case 'E': ellisys_server = optarg; ellisys_port = 24352; break; case 'P': use_pager = false; break; case 'J': jlink = optarg; break; case 'R': rtt = optarg; break; case 'C': set_default_pager_num_columns(atoi(optarg)); break; case 'c': if (strcmp("always", optarg) == 0) set_monitor_color(COLOR_ALWAYS); else if (strcmp("never", optarg) == 0) set_monitor_color(COLOR_NEVER); else if (strcmp("auto", optarg) == 0) set_monitor_color(COLOR_AUTO); else { fprintf(stderr, "Color option must be one of " "auto/always/never\n"); return EXIT_FAILURE; } break; case '#': packet_todo(); lmp_todo(); return EXIT_SUCCESS; case 'v': printf("%s\n", VERSION); return EXIT_SUCCESS; case 'h': usage(); return EXIT_SUCCESS; default: return EXIT_FAILURE; } } if (argc - optind > 0) { fprintf(stderr, "Invalid command line parameters\n"); return EXIT_FAILURE; } if (reader_path && analyze_path) { fprintf(stderr, "Display and analyze can't be combined\n"); return EXIT_FAILURE; } printf("Bluetooth monitor ver %s\n", VERSION); keys_setup(); packet_set_filter(filter_mask); if (analyze_path) { analyze_trace(analyze_path); return EXIT_SUCCESS; } if (reader_path) { if (ellisys_server) ellisys_enable(ellisys_server, ellisys_port); control_reader(reader_path, use_pager); return EXIT_SUCCESS; } if (writer_path && !control_writer(writer_path)) { printf("Failed to open '%s'\n", writer_path); return EXIT_FAILURE; } if (ellisys_server) ellisys_enable(ellisys_server, ellisys_port); if (!tty && !jlink && control_tracing() < 0) return EXIT_FAILURE; if (tty && control_tty(tty, tty_speed) < 0) return EXIT_FAILURE; if (jlink && control_rtt(jlink, rtt) < 0) return EXIT_FAILURE; exit_status = mainloop_run_with_signal(signal_callback, NULL); keys_cleanup(); return exit_status; }