| /* |
| * |
| * BlueZ - Bluetooth protocol stack for Linux |
| * |
| * Copyright (C) 2011-2014 Intel Corporation |
| * Copyright (C) 2002-2010 Marcel Holtmann <marcel@holtmann.org> |
| * |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
| * |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include <config.h> |
| #endif |
| |
| #include <stdio.h> |
| #include <ctype.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <getopt.h> |
| #include <sys/un.h> |
| |
| #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" |
| |
| 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 <file> Read traces in btsnoop format\n" |
| "\t-w, --write <file> Save traces in btsnoop format\n" |
| "\t-a, --analyze <file> Analyze traces in btsnoop format\n" |
| "\t-s, --server <socket> Start monitor server socket\n" |
| "\t-p, --priority <level> Show only priority or lower\n" |
| "\t-i, --index <num> Show only specified controller\n" |
| "\t-d, --tty <tty> Read data from TTY\n" |
| "\t-B, --tty-speed <rate> Set TTY speed (default 115200)\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-h, --help Show help options\n"); |
| } |
| |
| static const struct option main_options[] = { |
| { "tty", required_argument, NULL, 'd' }, |
| { "tty-speed", required_argument, NULL, 'B' }, |
| { "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' }, |
| { "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' }, |
| { "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; |
| 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; |
| int exit_status; |
| sigset_t mask; |
| |
| mainloop_init(); |
| |
| filter_mask |= PACKET_FILTER_SHOW_TIME_OFFSET; |
| |
| for (;;) { |
| int opt; |
| struct sockaddr_un addr; |
| |
| opt = getopt_long(argc, argv, "d:r:w:a:s:p:i:tTSAE:vh", |
| main_options, NULL); |
| if (opt < 0) |
| break; |
| |
| switch (opt) { |
| 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 '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 '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 '#': |
| 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; |
| } |
| |
| sigemptyset(&mask); |
| sigaddset(&mask, SIGINT); |
| sigaddset(&mask, SIGTERM); |
| |
| mainloop_set_signal(&mask, signal_callback, NULL, NULL); |
| |
| 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); |
| 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 && control_tracing() < 0) |
| return EXIT_FAILURE; |
| |
| if (tty && control_tty(tty, tty_speed) < 0) |
| return EXIT_FAILURE; |
| |
| exit_status = mainloop_run(); |
| |
| keys_cleanup(); |
| |
| return exit_status; |
| } |