123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305 |
- /*
- * Copyright (c) 2011 EIA Electronics
- *
- * Authors:
- * Kurt Van Dijck <kurt.van.dijck@eia.be>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the version 2 of the GNU General Public License
- * as published by the Free Software Foundation
- */
- #include <string.h>
- #include <time.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include <errno.h>
- #include <inttypes.h>
- #include <unistd.h>
- #include <getopt.h>
- #include <error.h>
- #include <sys/socket.h>
- #include <sys/ioctl.h>
- #include <sys/time.h>
- #include "libj1939.h"
- /*
- * getopt
- */
- static const char help_msg[] =
- "jspy: An SAE J1939 spy utility" "\n"
- "Usage: jspy [OPTION...] [[IFACE:][NAME|SA][,PGN]]" "\n"
- "\n"
- " -v, --verbose Increase verbosity" "\n"
- " -P, --promisc Run in promiscuous mode" "\n"
- " (= receive traffic not for this ECU)" "\n"
- " -b, --block=SIZE Use a receive buffer of SIZE (default 1024)" "\n"
- " -t, --time[=a|d|z|A] Show time: (a)bsolute, (d)elta, (z)ero, (A)bsolute w date" "\n"
- ;
- #ifdef _GNU_SOURCE
- static struct option long_opts[] = {
- { "help", no_argument, NULL, '?', },
- { "verbose", no_argument, NULL, 'v', },
- { "promisc", no_argument, NULL, 'P', },
- { "block", required_argument, NULL, 'b', },
- { "time", optional_argument, NULL, 't', },
- { },
- };
- #else
- #define getopt_long(argc, argv, optstring, longopts, longindex) \
- getopt((argc), (argv), (optstring))
- #endif
- static const char optstring[] = "vPb:t::?";
- /*
- * static variables
- */
- static struct {
- int verbose;
- struct sockaddr_can addr;
- int promisc;
- int time;
- int pkt_len;
- } s = {
- .pkt_len = 1024,
- .addr.can_addr.j1939 = {
- .name = J1939_NO_NAME,
- .addr = J1939_NO_ADDR,
- .pgn = J1939_NO_PGN,
- },
- };
- /*
- * usefull buffers
- */
- static const int ival_1 = 1;
- static char ctrlmsg[
- CMSG_SPACE(sizeof(struct timeval))
- + CMSG_SPACE(sizeof(uint8_t)) /* dest addr */
- + CMSG_SPACE(sizeof(uint64_t)) /* dest name */
- + CMSG_SPACE(sizeof(uint8_t)) /* priority */
- ];
- static struct iovec iov;
- static struct msghdr msg;
- static struct cmsghdr *cmsg;
- static uint8_t *buf;
- /*
- * program
- */
- int main(int argc, char **argv)
- {
- int ret, sock, j, opt;
- unsigned int len;
- struct timeval tref, tdut, ttmp;
- struct sockaddr_can src;
- struct j1939_filter filt;
- int filter = 0;
- uint8_t priority, dst_addr;
- uint64_t dst_name;
- long recvflags;
- #ifdef _GNU_SOURCE
- program_invocation_name = program_invocation_short_name;
- #endif
- /* argument parsing */
- while ((opt = getopt_long(argc, argv, optstring, long_opts, NULL)) != -1)
- switch (opt) {
- case 'v':
- ++s.verbose;
- break;
- case 'b':
- s.pkt_len = strtoul(optarg, 0, 0);
- break;
- case 'P':
- ++s.promisc;
- break;
- case 't':
- if (optarg) {
- if (!strchr("adzA", optarg[0]))
- error(1, 0, "unknown time option '%c'", optarg[0]);
- s.time = optarg[0];
- } else {
- s.time = 'z';
- }
- break;
- default:
- fputs(help_msg, stderr);
- exit(1);
- break;
- }
- if (argv[optind]) {
- optarg = argv[optind];
- ret = libj1939_str2addr(optarg, 0, &s.addr);
- if (ret < 0) {
- error(0, 0, "bad URI %s", optarg);
- return 1;
- }
- }
- buf = malloc(s.pkt_len);
- if (!buf)
- error(1, errno, "malloc %u", s.pkt_len);
- /* setup socket */
- sock = socket(PF_CAN, SOCK_DGRAM, CAN_J1939);
- if (sock < 0)
- error(1, errno, "socket(can, dgram, j1939)");
- memset(&filt, 0, sizeof(filt));
- if (s.addr.can_addr.j1939.name) {
- filt.name = s.addr.can_addr.j1939.name;
- filt.name_mask = ~0ULL;
- ++filter;
- }
- if (s.addr.can_addr.j1939.addr < 0xff) {
- filt.addr = s.addr.can_addr.j1939.addr;
- filt.addr_mask = ~0;
- ++filter;
- }
- if (s.addr.can_addr.j1939.pgn <= 0x3ffff) {
- filt.pgn = s.addr.can_addr.j1939.pgn;
- filt.pgn_mask = ~0;
- ++filter;
- }
- if (filter) {
- ret = setsockopt(sock, SOL_CAN_J1939, SO_J1939_FILTER, &filt, sizeof(filt));
- if (ret < 0)
- error(1, errno, "setsockopt filter");
- }
- if (s.promisc) {
- ret = setsockopt(sock, SOL_CAN_J1939, SO_J1939_PROMISC, &ival_1, sizeof(ival_1));
- if (ret < 0)
- error(1, errno, "setsockopt promisc");
- }
- if (s.time) {
- ret = setsockopt(sock, SOL_SOCKET, SO_TIMESTAMP, &ival_1, sizeof(ival_1));
- if (ret < 0)
- error(1, errno, "setsockopt timestamp");
- }
- ret = setsockopt(sock, SOL_SOCKET, SO_RCVBUF, &s.pkt_len, sizeof(s.pkt_len));
- if (ret < 0)
- error(1, errno, "setsockopt rcvbuf %u", s.pkt_len);
- /* bind(): to default, only ifindex is used. */
- memset(&src, 0, sizeof(src));
- src.can_ifindex = s.addr.can_ifindex;
- src.can_family = AF_CAN;
- src.can_addr.j1939.name = J1939_NO_NAME;
- src.can_addr.j1939.addr = J1939_NO_ADDR;
- src.can_addr.j1939.pgn = J1939_NO_PGN;
- ret = bind(sock, (void *)&src, sizeof(src));
- if (ret < 0)
- error(1, errno, "bind(%s)", argv[1]);
- /* these settings are static and can be held out of the hot path */
- iov.iov_base = &buf[0];
- msg.msg_name = &src;
- msg.msg_iov = &iov;
- msg.msg_iovlen = 1;
- msg.msg_control = &ctrlmsg;
- memset(&tref, 0, sizeof(tref));
- if (s.verbose)
- error(0, 0, "listening");
- while (1) {
- /* these settings may be modified by recvmsg() */
- iov.iov_len = s.pkt_len;
- msg.msg_namelen = sizeof(src);
- msg.msg_controllen = sizeof(ctrlmsg);
- msg.msg_flags = 0;
- ret = recvmsg(sock, &msg, 0);
- //ret = recvfrom(buf, s.pkt_len, 0, (void *)&addr, &len);
- if (ret < 0) {
- switch (errno) {
- case ENETDOWN:
- error(0, errno, "ifindex %i", s.addr.can_ifindex);
- continue;
- case EINTR:
- continue;
- default:
- error(1, errno, "recvmsg(ifindex %i)", s.addr.can_ifindex);
- break;
- }
- }
- len = ret;
- recvflags = 0;
- dst_addr = 0;
- priority = 0;
- for (cmsg = CMSG_FIRSTHDR(&msg); cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg)) {
- switch (cmsg->cmsg_level) {
- case SOL_SOCKET:
- if (cmsg->cmsg_type == SCM_TIMESTAMP) {
- memcpy(&tdut, CMSG_DATA(cmsg), sizeof(tdut));
- recvflags |= 1 << cmsg->cmsg_type;
- }
- break;
- case SOL_CAN_J1939:
- recvflags |= 1 << cmsg->cmsg_type;
- if (cmsg->cmsg_type == SCM_J1939_DEST_ADDR)
- dst_addr = *CMSG_DATA(cmsg);
- else if (cmsg->cmsg_type == SCM_J1939_DEST_NAME)
- memcpy(&dst_name, CMSG_DATA(cmsg), cmsg->cmsg_len - CMSG_LEN(0));
- else if (cmsg->cmsg_type == SCM_J1939_PRIO)
- priority = *CMSG_DATA(cmsg);
- break;
- }
- }
- if (recvflags & (1 << SCM_TIMESTAMP)) {
- if ('z' == s.time) {
- if (!tref.tv_sec)
- tref = tdut;
- timersub(&tdut, &tref, &ttmp);
- tdut = ttmp;
- goto abs_time;
- } else if ('d' == s.time) {
- timersub(&tdut, &tref, &ttmp);
- tref = tdut;
- tdut = ttmp;
- goto abs_time;
- } else if ('a' == s.time) {
- abs_time:
- printf("(%lu.%04lu)", tdut.tv_sec, tdut.tv_usec / 100);
- } else if ('A' == s.time) {
- struct tm tm;
- tm = *localtime(&tdut.tv_sec);
- printf("(%04u%02u%02uT%02u%02u%02u.%04lu)",
- tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday,
- tm.tm_hour, tm.tm_min, tm.tm_sec,
- tdut.tv_usec/100);
- }
- }
- printf(" %s ", libj1939_addr2str(&src));
- if (recvflags & (1 << SCM_J1939_DEST_NAME))
- printf("%016llx ", (unsigned long long)dst_name);
- else if (recvflags & (1 << SCM_J1939_DEST_ADDR))
- printf("%02x ", dst_addr);
- else
- printf("- ");
- printf("!%u ", priority);
- printf("[%i%s]", len, (msg.msg_flags & MSG_TRUNC) ? "..." : "");
- for (j = 0; j < len; ) {
- int end = j + 4;
- if (end > len)
- end = len;
- printf(" ");
- for (; j < end; ++j)
- printf("%02x", buf[j]);
- }
- printf("\n");
- }
- free(buf);
- return 0;
- }
|