__u8 rxpad_content; /* set content of padding byte (rx) */
/* __u8 value : content on rx path */
+
+ __u8 rx_ext_address; /* set address for extended addressing */
+ /* __u8 value : extended address (rx) */
};
struct can_isotp_fc_options {
#define CAN_ISOTP_HALF_DUPLEX 0x040 /* half duplex error state handling */
#define CAN_ISOTP_FORCE_TXSTMIN 0x080 /* ignore stmin from received FC */
#define CAN_ISOTP_FORCE_RXSTMIN 0x100 /* ignore CFs depending on rx stmin */
+#define CAN_ISOTP_RX_EXT_ADDR 0x200 /* different rx extended addressing */
/* default values */
fprintf(stderr, "Options: -s <can_id> (source can_id. Use 8 digits for extended IDs)\n");
fprintf(stderr, " -d <can_id> (destination can_id. Use 8 digits for extended IDs)\n");
fprintf(stderr, " -x <addr> (extended addressing mode. Use 'any' for all addresses)\n");
+ fprintf(stderr, " -X <addr> (extended addressing mode (rx addr). Use 'any' for all)\n");
fprintf(stderr, " -c (color mode)\n");
fprintf(stderr, " -a (print data also in ASCII-chars)\n");
fprintf(stderr, " -t <type> (timestamp: (a)bsolute/(d)elta/(z)ero/(A)bsolute w date)\n");
int ext = 0;
int extaddr = 0;
int extany = 0;
+ int rx_ext = 0;
+ int rx_extaddr = 0;
+ int rx_extany = 0;
int asc = 0;
int color = 0;
int timestamp = 0;
last_tv.tv_sec = 0;
last_tv.tv_usec = 0;
- while ((opt = getopt(argc, argv, "s:d:ax:ct:?")) != -1) {
+ while ((opt = getopt(argc, argv, "s:d:ax:X:ct:?")) != -1) {
switch (opt) {
case 's':
src = strtoul(optarg, (char **)NULL, 16);
extany = 1;
else
extaddr = strtoul(optarg, (char **)NULL, 16) & 0xFF;
+ break;
+ case 'X':
+ rx_ext = 1;
+ if (!strncmp(optarg, "any", 3))
+ rx_extany = 1;
+ else
+ rx_extaddr = strtoul(optarg, (char **)NULL, 16) & 0xFF;
break;
case 't':
}
}
+ if (rx_ext && !ext) {
+ print_usage(basename(argv[0]));
+ exit(0);
+ }
+
if ((argc - optind) != 1 || src == NO_CAN_ID || dst == NO_CAN_ID) {
print_usage(basename(argv[0]));
exit(0);
return 1;
} else {
- if (ext && !extany && extaddr != frame.data[0])
+ if (frame.can_id == src && ext && !extany && extaddr != frame.data[0])
+ continue;
+
+ if (frame.can_id == dst && rx_ext && !rx_extany && rx_extaddr != frame.data[0])
continue;
if (color)
fprintf(stderr, "Options: -s <can_id> (source can_id. Use 8 digits for extended IDs)\n");
fprintf(stderr, " -d <can_id> (destination can_id. Use 8 digits for extended IDs)\n");
fprintf(stderr, " -x <addr> (extended addressing mode.)\n");
+ fprintf(stderr, " -X <addr> (extended addressing mode (rx addr).)\n");
fprintf(stderr, " -p <byte> (set and enable padding byte)\n");
fprintf(stderr, " -P <mode> (check padding in SF/CF. (l)ength (c)ontent (a)ll)\n");
fprintf(stderr, " -b <bs> (blocksize. 0 = off)\n");
addr.can_addr.tp.tx_id = addr.can_addr.tp.rx_id = NO_CAN_ID;
- while ((opt = getopt(argc, argv, "s:d:x:p:P:b:m:w:f:l?")) != -1) {
+ while ((opt = getopt(argc, argv, "s:d:x:X:p:P:b:m:w:f:l?")) != -1) {
switch (opt) {
case 's':
addr.can_addr.tp.tx_id = strtoul(optarg, (char **)NULL, 16);
opts.ext_address = strtoul(optarg, (char **)NULL, 16) & 0xFF;
break;
+ case 'X':
+ opts.flags |= CAN_ISOTP_RX_EXT_ADDR;
+ opts.rx_ext_address = strtoul(optarg, (char **)NULL, 16) & 0xFF;
+ break;
+
case 'p':
opts.flags |= CAN_ISOTP_RX_PADDING;
opts.rxpad_content = strtoul(optarg, (char **)NULL, 16) & 0xFF;
exit(1);
}
+ if ((opts.flags & CAN_ISOTP_RX_EXT_ADDR) && (!(opts.flags & CAN_ISOTP_EXTEND_ADDR))) {
+ print_usage(basename(argv[0]));
+ exit(1);
+ }
+
if ((s = socket(PF_CAN, SOCK_DGRAM, CAN_ISOTP)) < 0) {
perror("socket");
exit(1);
fprintf(stderr, "\nUsage: %s [options] <CAN interface>\n", prg);
fprintf(stderr, "Options: -s <can_id> (source can_id. Use 8 digits for extended IDs)\n");
fprintf(stderr, " -d <can_id> (destination can_id. Use 8 digits for extended IDs)\n");
- fprintf(stderr, " -x <addr> (extended addressing mode. Use 'any' for all addresses)\n");
+ fprintf(stderr, " -x <addr> (extended addressing mode)\n");
+ fprintf(stderr, " -X <addr> (extended addressing mode - rx addr)\n");
fprintf(stderr, " -p <byte> (set and enable padding byte)\n");
fprintf(stderr, " -P <mode> (check padding in FC. (l)ength (c)ontent (a)ll)\n");
fprintf(stderr, " -t <time ns> (frame transmit time (N_As) in nanosecs)\n");
addr.can_addr.tp.tx_id = addr.can_addr.tp.rx_id = NO_CAN_ID;
- while ((opt = getopt(argc, argv, "s:d:x:p:P:t:f:D:?")) != -1) {
+ while ((opt = getopt(argc, argv, "s:d:x:X:p:P:t:f:D:?")) != -1) {
switch (opt) {
case 's':
addr.can_addr.tp.tx_id = strtoul(optarg, (char **)NULL, 16);
opts.ext_address = strtoul(optarg, (char **)NULL, 16) & 0xFF;
break;
+ case 'X':
+ opts.flags |= CAN_ISOTP_RX_EXT_ADDR;
+ opts.rx_ext_address = strtoul(optarg, (char **)NULL, 16) & 0xFF;
+ break;
+
case 'p':
opts.flags |= CAN_ISOTP_TX_PADDING;
opts.txpad_content = strtoul(optarg, (char **)NULL, 16) & 0xFF;
exit(1);
}
+ if ((opts.flags & CAN_ISOTP_RX_EXT_ADDR) && (!(opts.flags & CAN_ISOTP_EXTEND_ADDR))) {
+ print_usage(basename(argv[0]));
+ exit(1);
+ }
+
if ((s = socket(PF_CAN, SOCK_DGRAM, CAN_ISOTP)) < 0) {
perror("socket");
exit(1);
fprintf(stderr, "\nUsage: %s [options] <CAN interface>\n", prg);
fprintf(stderr, "Options: -s <can_id> (source can_id. Use 8 digits for extended IDs)\n");
fprintf(stderr, " -d <can_id> (destination can_id. Use 8 digits for extended IDs)\n");
- fprintf(stderr, " -x <addr> (extended addressing mode.)\n");
+ fprintf(stderr, " -x <addr> (extended addressing mode)\n");
+ fprintf(stderr, " -X <addr> (extended addressing mode - rx addr)\n");
fprintf(stderr, " -c (color mode)\n");
fprintf(stderr, " -t <type> (timestamp: (a)bsolute/(d)elta/(z)ero/(A)bsolute w date)\n");
fprintf(stderr, " -f <format> (1 = HEX, 2 = ASCII, 3 = HEX & ASCII - default: %d)\n", FORMAT_DEFAULT);
unsigned char buffer[4096];
int nbytes;
- while ((opt = getopt(argc, argv, "s:d:x:h:ct:f:?")) != -1) {
+ while ((opt = getopt(argc, argv, "s:d:x:X:h:ct:f:?")) != -1) {
switch (opt) {
case 's':
src = strtoul(optarg, (char **)NULL, 16);
opts.ext_address = strtoul(optarg, (char **)NULL, 16) & 0xFF;
break;
+ case 'X':
+ opts.flags |= CAN_ISOTP_RX_EXT_ADDR;
+ opts.rx_ext_address = strtoul(optarg, (char **)NULL, 16) & 0xFF;
+ break;
+
case 'f':
format = (atoi(optarg) & (FORMAT_ASCII | FORMAT_HEX));
break;
exit(1);
}
+ if ((opts.flags & CAN_ISOTP_RX_EXT_ADDR) && (!(opts.flags & CAN_ISOTP_EXTEND_ADDR))) {
+ print_usage(basename(argv[0]));
+ exit(1);
+ }
+
if ((s = socket(PF_CAN, SOCK_DGRAM, CAN_ISOTP)) < 0) {
perror("socket");
exit(1);
opts.flags |= CAN_ISOTP_LISTEN_MODE;
- setsockopt(s, SOL_CAN_ISOTP, CAN_ISOTP_OPTS, &opts, sizeof(opts));
- setsockopt(t, SOL_CAN_ISOTP, CAN_ISOTP_OPTS, &opts, sizeof(opts));
-
addr.can_family = AF_CAN;
strcpy(ifr.ifr_name, argv[optind]);
ioctl(s, SIOCGIFINDEX, &ifr);
addr.can_ifindex = ifr.ifr_ifindex;
- addr.can_addr.tp.tx_id = dst;
- addr.can_addr.tp.rx_id = src;
+ setsockopt(s, SOL_CAN_ISOTP, CAN_ISOTP_OPTS, &opts, sizeof(opts));
+
+ addr.can_addr.tp.tx_id = src;
+ addr.can_addr.tp.rx_id = dst;
if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
perror("bind");
exit(1);
}
- addr.can_addr.tp.tx_id = src;
- addr.can_addr.tp.rx_id = dst;
+ if (opts.flags & CAN_ISOTP_RX_EXT_ADDR) {
+ /* flip extended address info due to separate rx ext addr */
+ __u8 tmpext;
+
+ tmpext = opts.ext_address;
+ opts.ext_address = opts.rx_ext_address;
+ opts.rx_ext_address = tmpext;
+ }
+
+ setsockopt(t, SOL_CAN_ISOTP, CAN_ISOTP_OPTS, &opts, sizeof(opts));
+
+ addr.can_addr.tp.tx_id = dst;
+ addr.can_addr.tp.rx_id = src;
if (bind(t, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
perror("bind");
}
if (nbytes > 4095)
return -1;
- printbuf(buffer, nbytes, color?1:0, timestamp, format,
- &tv, &last_tv, src, s, ifr.ifr_name, head);
+ printbuf(buffer, nbytes, color?2:0, timestamp, format,
+ &tv, &last_tv, dst, s, ifr.ifr_name, head);
}
if (FD_ISSET(t, &rdfs)) {
}
if (nbytes > 4095)
return -1;
- printbuf(buffer, nbytes, color?2:0, timestamp, format,
- &tv, &last_tv, dst, t, ifr.ifr_name, head);
+ printbuf(buffer, nbytes, color?1:0, timestamp, format,
+ &tv, &last_tv, src, t, ifr.ifr_name, head);
}
}