Well that flipping works. Good night!
This commit is contained in:
parent
8074d19427
commit
66f4990a1e
6 changed files with 169 additions and 27 deletions
|
@ -7,7 +7,7 @@ INCLUDE_PATH = ../include
|
||||||
CFLAGS += -I$(INCLUDE_PATH)
|
CFLAGS += -I$(INCLUDE_PATH)
|
||||||
LDFLAGS = -L../src -lhexagram
|
LDFLAGS = -L../src -lhexagram
|
||||||
|
|
||||||
EXAMPLES = convert pcapread
|
EXAMPLES = convert pcapread replay
|
||||||
|
|
||||||
RM = /bin/rm
|
RM = /bin/rm
|
||||||
|
|
||||||
|
|
|
@ -108,7 +108,7 @@ static ssize_t handle_block_packet(hexagram_pcapng_stream *stream,
|
||||||
total += readlen;
|
total += readlen;
|
||||||
}
|
}
|
||||||
|
|
||||||
usec = (uint64_t)header.timestamp[0] << 32
|
usec = ((uint64_t)header.timestamp[0] << 32)
|
||||||
| (uint64_t)header.timestamp[1];
|
| (uint64_t)header.timestamp[1];
|
||||||
|
|
||||||
timestamp.tv_sec = usec / 1000000;
|
timestamp.tv_sec = usec / 1000000;
|
||||||
|
|
|
@ -151,7 +151,7 @@ static ssize_t handle_block_packet(hexagram_pcapng_stream *stream,
|
||||||
((struct can_frame *)&body)->can_id,
|
((struct can_frame *)&body)->can_id,
|
||||||
((struct can_frame *)&body)->can_dlc);
|
((struct can_frame *)&body)->can_dlc);
|
||||||
|
|
||||||
if (write(data->sock, (struct can_frame *)body, sizeof(struct can_frame)) < 0) {
|
if (write(data->sock, body, sizeof(struct can_frame)) < 0) {
|
||||||
goto error_io;
|
goto error_io;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
131
examples/replay.c
Normal file
131
examples/replay.c
Normal file
|
@ -0,0 +1,131 @@
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <inttypes.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
|
||||||
|
#include <net/if.h>
|
||||||
|
#include <linux/can.h>
|
||||||
|
#include <linux/can/raw.h>
|
||||||
|
|
||||||
|
#include <hexagram/capture.h>
|
||||||
|
|
||||||
|
static void usage(int argc, char **argv, const char *message, ...) {
|
||||||
|
if (message) {
|
||||||
|
va_list args;
|
||||||
|
|
||||||
|
va_start(args, message);
|
||||||
|
vfprintf(stderr, message, args);
|
||||||
|
va_end(args);
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "usage: %s canif [file.can]\n", argv[0]);
|
||||||
|
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int capture_replay(hexagram_capture *capture, int sock) {
|
||||||
|
ssize_t len;
|
||||||
|
uint32_t timestamp[2],
|
||||||
|
usec_last = 0;
|
||||||
|
struct can_frame frame;
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
if ((len = hexagram_capture_read(capture,
|
||||||
|
×tamp[0],
|
||||||
|
×tamp[1],
|
||||||
|
&frame)) < 0) {
|
||||||
|
goto error_io;
|
||||||
|
} else if (len == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (usec_last) {
|
||||||
|
usleep((useconds_t)(timestamp[1] - usec_last));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (write(sock, &frame, sizeof(struct can_frame)) < 0) {
|
||||||
|
goto error_io;
|
||||||
|
}
|
||||||
|
|
||||||
|
usec_last = timestamp[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
error_io:
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char **argv) {
|
||||||
|
struct sockaddr_can addr;
|
||||||
|
struct ifreq ifr;
|
||||||
|
int sock;
|
||||||
|
|
||||||
|
hexagram_capture *capture;
|
||||||
|
|
||||||
|
if (argc == 2) {
|
||||||
|
if ((capture = hexagram_capture_open_fd(fileno(stdin), O_RDONLY)) == NULL) {
|
||||||
|
goto error_capture_open;
|
||||||
|
}
|
||||||
|
} else if (argc == 3) {
|
||||||
|
if ((capture = hexagram_capture_open_file(argv[2], O_RDONLY)) == NULL) {
|
||||||
|
perror("hexagram_capture_open_file()");
|
||||||
|
|
||||||
|
goto error_capture_open;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
usage(argc, argv, NULL);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((sock = socket(PF_CAN, SOCK_RAW, CAN_RAW)) < 0) {
|
||||||
|
perror("socket()");
|
||||||
|
|
||||||
|
goto error_socket;
|
||||||
|
}
|
||||||
|
|
||||||
|
strcpy(ifr.ifr_name, argv[1]);
|
||||||
|
|
||||||
|
ioctl(sock, SIOCGIFINDEX, &ifr);
|
||||||
|
|
||||||
|
addr.can_family = AF_CAN;
|
||||||
|
addr.can_ifindex = ifr.ifr_ifindex;
|
||||||
|
|
||||||
|
if (bind(sock, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
|
||||||
|
perror("bind()");
|
||||||
|
|
||||||
|
goto error_bind;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (capture_replay(capture, sock) < 0) {
|
||||||
|
perror("capture_replay()");
|
||||||
|
|
||||||
|
goto error_capture_replay;
|
||||||
|
}
|
||||||
|
|
||||||
|
close(sock);
|
||||||
|
|
||||||
|
if (argc == 3) {
|
||||||
|
hexagram_capture_close(capture);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
error_capture_replay:
|
||||||
|
error_bind:
|
||||||
|
close(sock);
|
||||||
|
|
||||||
|
error_socket:
|
||||||
|
if (argc == 3) {
|
||||||
|
hexagram_capture_close(capture);
|
||||||
|
}
|
||||||
|
|
||||||
|
error_capture_open:
|
||||||
|
return 1;
|
||||||
|
}
|
|
@ -29,12 +29,14 @@ hexagram_capture *hexagram_capture_open_file(const char *file, int flags);
|
||||||
|
|
||||||
void hexagram_capture_destroy(hexagram_capture *capture);
|
void hexagram_capture_destroy(hexagram_capture *capture);
|
||||||
|
|
||||||
int hexagram_capture_read(hexagram_capture *capture,
|
void hexagram_capture_close(hexagram_capture *capture);
|
||||||
|
|
||||||
|
ssize_t hexagram_capture_read(hexagram_capture *capture,
|
||||||
uint32_t *timestamp_hi,
|
uint32_t *timestamp_hi,
|
||||||
uint32_t *timestamp_lo,
|
uint32_t *timestamp_lo,
|
||||||
struct can_frame *frame);
|
struct can_frame *frame);
|
||||||
|
|
||||||
int hexagram_capture_write(hexagram_capture *capture,
|
ssize_t hexagram_capture_write(hexagram_capture *capture,
|
||||||
struct can_frame *frame,
|
struct can_frame *frame,
|
||||||
struct timeval *timestamp);
|
struct timeval *timestamp);
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <inttypes.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
@ -20,7 +21,7 @@ hexagram_capture *hexagram_capture_open_fd(int fd, int flags) {
|
||||||
goto error_malloc;
|
goto error_malloc;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags & O_WRONLY && !(flags & O_APPEND)) {
|
if ((flags & O_WRONLY) && !(flags & O_APPEND)) {
|
||||||
memcpy(header.magic, HEXAGRAM_CAPTURE_MAGIC, strlen(HEXAGRAM_CAPTURE_MAGIC));
|
memcpy(header.magic, HEXAGRAM_CAPTURE_MAGIC, strlen(HEXAGRAM_CAPTURE_MAGIC));
|
||||||
|
|
||||||
header.endian = HEXAGRAM_CAPTURE_ENDIAN;
|
header.endian = HEXAGRAM_CAPTURE_ENDIAN;
|
||||||
|
@ -28,7 +29,7 @@ hexagram_capture *hexagram_capture_open_fd(int fd, int flags) {
|
||||||
if (write(fd, &header, sizeof(header)) < 0) {
|
if (write(fd, &header, sizeof(header)) < 0) {
|
||||||
goto error_io;
|
goto error_io;
|
||||||
}
|
}
|
||||||
} else if (flags & O_RDONLY) {
|
} else if (flags == O_RDONLY) {
|
||||||
ssize_t readlen;
|
ssize_t readlen;
|
||||||
|
|
||||||
if ((readlen = read(fd, &header, sizeof(header))) < 0) {
|
if ((readlen = read(fd, &header, sizeof(header))) < 0) {
|
||||||
|
@ -80,7 +81,13 @@ void hexagram_capture_destroy(hexagram_capture *capture) {
|
||||||
free(capture);
|
free(capture);
|
||||||
}
|
}
|
||||||
|
|
||||||
int hexagram_capture_read(hexagram_capture *capture,
|
void hexagram_capture_close(hexagram_capture *capture) {
|
||||||
|
close(capture->fd);
|
||||||
|
|
||||||
|
hexagram_capture_destroy(capture);
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t hexagram_capture_read(hexagram_capture *capture,
|
||||||
uint32_t *timestamp_hi,
|
uint32_t *timestamp_hi,
|
||||||
uint32_t *timestamp_lo,
|
uint32_t *timestamp_lo,
|
||||||
struct can_frame *frame) {
|
struct can_frame *frame) {
|
||||||
|
@ -93,21 +100,22 @@ int hexagram_capture_read(hexagram_capture *capture,
|
||||||
goto error_io;
|
goto error_io;
|
||||||
}
|
}
|
||||||
|
|
||||||
*timestamp_hi = data.timestamp_hi;
|
if (timestamp_hi) *timestamp_hi = data.timestamp_hi;
|
||||||
*timestamp_lo = data.timestamp_lo;
|
if (timestamp_lo) *timestamp_lo = data.timestamp_lo;
|
||||||
|
|
||||||
memcpy(frame, &data.frame, sizeof(data.frame));
|
memcpy(frame, &data.frame, sizeof(data.frame));
|
||||||
|
|
||||||
return 0;
|
return len;
|
||||||
|
|
||||||
error_io:
|
error_io:
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int hexagram_capture_write(hexagram_capture *capture,
|
ssize_t hexagram_capture_write(hexagram_capture *capture,
|
||||||
struct can_frame *frame,
|
struct can_frame *frame,
|
||||||
struct timeval *timestamp) {
|
struct timeval *timestamp) {
|
||||||
hexagram_capture_frame data;
|
hexagram_capture_frame data;
|
||||||
|
uint64_t usec;
|
||||||
|
|
||||||
if (timestamp == NULL) {
|
if (timestamp == NULL) {
|
||||||
struct timeval now;
|
struct timeval now;
|
||||||
|
@ -116,13 +124,14 @@ int hexagram_capture_write(hexagram_capture *capture,
|
||||||
goto error_gettimeofday;
|
goto error_gettimeofday;
|
||||||
}
|
}
|
||||||
|
|
||||||
data.timestamp_hi = now.tv_usec & 0xffffffff00000000 >> 32;
|
usec = now.tv_usec + now.tv_sec * 1000000;
|
||||||
data.timestamp_lo = now.tv_usec & 0x00000000ffffffff;
|
|
||||||
} else {
|
} else {
|
||||||
data.timestamp_hi = timestamp->tv_usec & 0xffffffff00000000 >> 32;
|
usec = timestamp->tv_usec + timestamp->tv_sec * 1000000;
|
||||||
data.timestamp_lo = timestamp->tv_usec & 0x00000000ffffffff;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
data.timestamp_hi = usec & 0xffffffff00000000 >> 32;
|
||||||
|
data.timestamp_lo = usec & 0x00000000ffffffff;
|
||||||
|
|
||||||
memcpy(&data.frame, frame, sizeof(data.frame));
|
memcpy(&data.frame, frame, sizeof(data.frame));
|
||||||
|
|
||||||
return write(capture->fd, &data, sizeof(data));
|
return write(capture->fd, &data, sizeof(data));
|
||||||
|
|
Loading…
Add table
Reference in a new issue