jspy.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. /*
  2. * Copyright (c) 2011 EIA Electronics
  3. *
  4. * Authors:
  5. * Kurt Van Dijck <kurt.van.dijck@eia.be>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the version 2 of the GNU General Public License
  9. * as published by the Free Software Foundation
  10. */
  11. #include <string.h>
  12. #include <time.h>
  13. #include <stdlib.h>
  14. #include <stdio.h>
  15. #include <errno.h>
  16. #include <inttypes.h>
  17. #include <unistd.h>
  18. #include <getopt.h>
  19. #include <error.h>
  20. #include <sys/socket.h>
  21. #include <sys/ioctl.h>
  22. #include <sys/time.h>
  23. #include "libj1939.h"
  24. /*
  25. * getopt
  26. */
  27. static const char help_msg[] =
  28. "jspy: An SAE J1939 spy utility" "\n"
  29. "Usage: jspy [OPTION...] [[IFACE:][NAME|SA][,PGN]]" "\n"
  30. "\n"
  31. " -v, --verbose Increase verbosity" "\n"
  32. " -P, --promisc Run in promiscuous mode" "\n"
  33. " (= receive traffic not for this ECU)" "\n"
  34. " -b, --block=SIZE Use a receive buffer of SIZE (default 1024)" "\n"
  35. " -t, --time[=a|d|z|A] Show time: (a)bsolute, (d)elta, (z)ero, (A)bsolute w date" "\n"
  36. ;
  37. #ifdef _GNU_SOURCE
  38. static struct option long_opts[] = {
  39. { "help", no_argument, NULL, '?', },
  40. { "verbose", no_argument, NULL, 'v', },
  41. { "promisc", no_argument, NULL, 'P', },
  42. { "block", required_argument, NULL, 'b', },
  43. { "time", optional_argument, NULL, 't', },
  44. { },
  45. };
  46. #else
  47. #define getopt_long(argc, argv, optstring, longopts, longindex) \
  48. getopt((argc), (argv), (optstring))
  49. #endif
  50. static const char optstring[] = "vPb:t::?";
  51. /*
  52. * static variables
  53. */
  54. static struct {
  55. int verbose;
  56. struct sockaddr_can addr;
  57. int promisc;
  58. int time;
  59. int pkt_len;
  60. } s = {
  61. .pkt_len = 1024,
  62. .addr.can_addr.j1939 = {
  63. .name = J1939_NO_NAME,
  64. .addr = J1939_NO_ADDR,
  65. .pgn = J1939_NO_PGN,
  66. },
  67. };
  68. /*
  69. * usefull buffers
  70. */
  71. static const int ival_1 = 1;
  72. static char ctrlmsg[
  73. CMSG_SPACE(sizeof(struct timeval))
  74. + CMSG_SPACE(sizeof(uint8_t)) /* dest addr */
  75. + CMSG_SPACE(sizeof(uint64_t)) /* dest name */
  76. + CMSG_SPACE(sizeof(uint8_t)) /* priority */
  77. ];
  78. static struct iovec iov;
  79. static struct msghdr msg;
  80. static struct cmsghdr *cmsg;
  81. static uint8_t *buf;
  82. /*
  83. * program
  84. */
  85. int main(int argc, char **argv)
  86. {
  87. int ret, sock, j, opt;
  88. unsigned int len;
  89. struct timeval tref, tdut, ttmp;
  90. struct sockaddr_can src;
  91. struct j1939_filter filt;
  92. int filter = 0;
  93. uint8_t priority, dst_addr;
  94. uint64_t dst_name;
  95. long recvflags;
  96. #ifdef _GNU_SOURCE
  97. program_invocation_name = program_invocation_short_name;
  98. #endif
  99. /* argument parsing */
  100. while ((opt = getopt_long(argc, argv, optstring, long_opts, NULL)) != -1)
  101. switch (opt) {
  102. case 'v':
  103. ++s.verbose;
  104. break;
  105. case 'b':
  106. s.pkt_len = strtoul(optarg, 0, 0);
  107. break;
  108. case 'P':
  109. ++s.promisc;
  110. break;
  111. case 't':
  112. if (optarg) {
  113. if (!strchr("adzA", optarg[0]))
  114. error(1, 0, "unknown time option '%c'", optarg[0]);
  115. s.time = optarg[0];
  116. } else {
  117. s.time = 'z';
  118. }
  119. break;
  120. default:
  121. fputs(help_msg, stderr);
  122. exit(1);
  123. break;
  124. }
  125. if (argv[optind]) {
  126. optarg = argv[optind];
  127. ret = libj1939_str2addr(optarg, 0, &s.addr);
  128. if (ret < 0) {
  129. error(0, 0, "bad URI %s", optarg);
  130. return 1;
  131. }
  132. }
  133. buf = malloc(s.pkt_len);
  134. if (!buf)
  135. error(1, errno, "malloc %u", s.pkt_len);
  136. /* setup socket */
  137. sock = socket(PF_CAN, SOCK_DGRAM, CAN_J1939);
  138. if (sock < 0)
  139. error(1, errno, "socket(can, dgram, j1939)");
  140. memset(&filt, 0, sizeof(filt));
  141. if (s.addr.can_addr.j1939.name) {
  142. filt.name = s.addr.can_addr.j1939.name;
  143. filt.name_mask = ~0ULL;
  144. ++filter;
  145. }
  146. if (s.addr.can_addr.j1939.addr < 0xff) {
  147. filt.addr = s.addr.can_addr.j1939.addr;
  148. filt.addr_mask = ~0;
  149. ++filter;
  150. }
  151. if (s.addr.can_addr.j1939.pgn <= 0x3ffff) {
  152. filt.pgn = s.addr.can_addr.j1939.pgn;
  153. filt.pgn_mask = ~0;
  154. ++filter;
  155. }
  156. if (filter) {
  157. ret = setsockopt(sock, SOL_CAN_J1939, SO_J1939_FILTER, &filt, sizeof(filt));
  158. if (ret < 0)
  159. error(1, errno, "setsockopt filter");
  160. }
  161. if (s.promisc) {
  162. ret = setsockopt(sock, SOL_CAN_J1939, SO_J1939_PROMISC, &ival_1, sizeof(ival_1));
  163. if (ret < 0)
  164. error(1, errno, "setsockopt promisc");
  165. }
  166. if (s.time) {
  167. ret = setsockopt(sock, SOL_SOCKET, SO_TIMESTAMP, &ival_1, sizeof(ival_1));
  168. if (ret < 0)
  169. error(1, errno, "setsockopt timestamp");
  170. }
  171. ret = setsockopt(sock, SOL_SOCKET, SO_RCVBUF, &s.pkt_len, sizeof(s.pkt_len));
  172. if (ret < 0)
  173. error(1, errno, "setsockopt rcvbuf %u", s.pkt_len);
  174. /* bind(): to default, only ifindex is used. */
  175. memset(&src, 0, sizeof(src));
  176. src.can_ifindex = s.addr.can_ifindex;
  177. src.can_family = AF_CAN;
  178. src.can_addr.j1939.name = J1939_NO_NAME;
  179. src.can_addr.j1939.addr = J1939_NO_ADDR;
  180. src.can_addr.j1939.pgn = J1939_NO_PGN;
  181. ret = bind(sock, (void *)&src, sizeof(src));
  182. if (ret < 0)
  183. error(1, errno, "bind(%s)", argv[1]);
  184. /* these settings are static and can be held out of the hot path */
  185. iov.iov_base = &buf[0];
  186. msg.msg_name = &src;
  187. msg.msg_iov = &iov;
  188. msg.msg_iovlen = 1;
  189. msg.msg_control = &ctrlmsg;
  190. memset(&tref, 0, sizeof(tref));
  191. if (s.verbose)
  192. error(0, 0, "listening");
  193. while (1) {
  194. /* these settings may be modified by recvmsg() */
  195. iov.iov_len = s.pkt_len;
  196. msg.msg_namelen = sizeof(src);
  197. msg.msg_controllen = sizeof(ctrlmsg);
  198. msg.msg_flags = 0;
  199. ret = recvmsg(sock, &msg, 0);
  200. //ret = recvfrom(buf, s.pkt_len, 0, (void *)&addr, &len);
  201. if (ret < 0) {
  202. switch (errno) {
  203. case ENETDOWN:
  204. error(0, errno, "ifindex %i", s.addr.can_ifindex);
  205. continue;
  206. case EINTR:
  207. continue;
  208. default:
  209. error(1, errno, "recvmsg(ifindex %i)", s.addr.can_ifindex);
  210. break;
  211. }
  212. }
  213. len = ret;
  214. recvflags = 0;
  215. dst_addr = 0;
  216. priority = 0;
  217. for (cmsg = CMSG_FIRSTHDR(&msg); cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg)) {
  218. switch (cmsg->cmsg_level) {
  219. case SOL_SOCKET:
  220. if (cmsg->cmsg_type == SCM_TIMESTAMP) {
  221. memcpy(&tdut, CMSG_DATA(cmsg), sizeof(tdut));
  222. recvflags |= 1 << cmsg->cmsg_type;
  223. }
  224. break;
  225. case SOL_CAN_J1939:
  226. recvflags |= 1 << cmsg->cmsg_type;
  227. if (cmsg->cmsg_type == SCM_J1939_DEST_ADDR)
  228. dst_addr = *CMSG_DATA(cmsg);
  229. else if (cmsg->cmsg_type == SCM_J1939_DEST_NAME)
  230. memcpy(&dst_name, CMSG_DATA(cmsg), cmsg->cmsg_len - CMSG_LEN(0));
  231. else if (cmsg->cmsg_type == SCM_J1939_PRIO)
  232. priority = *CMSG_DATA(cmsg);
  233. break;
  234. }
  235. }
  236. if (recvflags & (1 << SCM_TIMESTAMP)) {
  237. if ('z' == s.time) {
  238. if (!tref.tv_sec)
  239. tref = tdut;
  240. timersub(&tdut, &tref, &ttmp);
  241. tdut = ttmp;
  242. goto abs_time;
  243. } else if ('d' == s.time) {
  244. timersub(&tdut, &tref, &ttmp);
  245. tref = tdut;
  246. tdut = ttmp;
  247. goto abs_time;
  248. } else if ('a' == s.time) {
  249. abs_time:
  250. printf("(%lu.%04lu)", tdut.tv_sec, tdut.tv_usec / 100);
  251. } else if ('A' == s.time) {
  252. struct tm tm;
  253. tm = *localtime(&tdut.tv_sec);
  254. printf("(%04u%02u%02uT%02u%02u%02u.%04lu)",
  255. tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday,
  256. tm.tm_hour, tm.tm_min, tm.tm_sec,
  257. tdut.tv_usec/100);
  258. }
  259. }
  260. printf(" %s ", libj1939_addr2str(&src));
  261. if (recvflags & (1 << SCM_J1939_DEST_NAME))
  262. printf("%016llx ", (unsigned long long)dst_name);
  263. else if (recvflags & (1 << SCM_J1939_DEST_ADDR))
  264. printf("%02x ", dst_addr);
  265. else
  266. printf("- ");
  267. printf("!%u ", priority);
  268. printf("[%i%s]", len, (msg.msg_flags & MSG_TRUNC) ? "..." : "");
  269. for (j = 0; j < len; ) {
  270. int end = j + 4;
  271. if (end > len)
  272. end = len;
  273. printf(" ");
  274. for (; j < end; ++j)
  275. printf("%02x", buf[j]);
  276. }
  277. printf("\n");
  278. }
  279. free(buf);
  280. return 0;
  281. }