#include #include #include #include #include #include #include #include #include #include #include #include #include #include extern int optind, opterr, optopt; static int s = -1; static int running = 1; enum { VERSION_OPTION = CHAR_MAX + 1, }; void print_usage(char *prg) { fprintf(stderr, "Usage: %s [Options]\n" "Options:\n" " -f, --family=FAMILY protocol family (default PF_CAN = %d)\n" " -t, --type=TYPE socket type, see man 2 socket (default SOCK_RAW = %d)\n" " -p, --protocol=PROTO CAN protocol (default CAN_PROTO_RAW = %d)\n" " -h, --help this help\n" " --version print version information and exit\n", prg, PF_CAN, SOCK_RAW, CAN_PROTO_RAW); } void sigterm(int signo) { running = 0; } int main(int argc, char **argv) { int family = PF_CAN, type = SOCK_RAW, proto = CAN_PROTO_RAW; int opt; struct sockaddr_can addr; struct can_frame frame; int nbytes, i; struct ifreq ifr; signal(SIGTERM, sigterm); signal(SIGHUP, sigterm); struct option long_options[] = { { "help", no_argument, 0, 'h' }, { "family", required_argument, 0, 'f' }, { "protocol", required_argument, 0, 'p' }, { "type", required_argument, 0, 't' }, { "version", no_argument, 0, VERSION_OPTION}, { 0, 0, 0, 0}, }; while ((opt = getopt_long(argc, argv, "f:t:p:", long_options, NULL)) != -1) { switch (opt) { case 'h': print_usage(basename(argv[0])); exit(0); case 'f': family = strtoul(optarg, NULL, 0); break; case 't': type = strtoul(optarg, NULL, 0); break; case 'p': proto = strtoul(optarg, NULL, 0); break; case VERSION_OPTION: printf("candump %s\n",VERSION); exit(0); default: fprintf(stderr, "Unknown option %c\n", opt); break; } } if (optind == argc) { print_usage(basename(argv[0])); exit(0); } printf("interface = %s, family = %d, type = %d, proto = %d\n", argv[optind], family, type, proto); if ((s = socket(family, type, proto)) < 0) { perror("socket"); return 1; } addr.can_family = family; strncpy(ifr.ifr_name, argv[optind], sizeof(ifr.ifr_name)); ioctl(s, SIOCGIFINDEX, &ifr); addr.can_ifindex = ifr.ifr_ifindex; addr.can_id = CAN_FLAG_ALL; if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) < 0) { perror("bind"); return 1; } while (running) { if ((nbytes = read(s, &frame, sizeof(struct can_frame))) < 0) { perror("read"); return 1; } else { if (frame.can_id & CAN_FLAG_EXTENDED) printf("<0x%08x> ", frame.can_id & CAN_ID_EXT_MASK); else printf("<0x%03x> ", frame.can_id & CAN_ID_STD_MASK); printf("[%d] ", frame.can_dlc); for (i = 0; i < frame.can_dlc; i++) { printf("%02x ", frame.payload.data_u8[i]); } if (frame.can_id & CAN_FLAG_RTR) printf("remote request"); printf("\n"); } } exit (EXIT_SUCCESS); }