2019-02-11 20:12:09 -06:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2019-02-12 00:00:13 -06:00
|
|
|
#include <inttypes.h>
|
2019-02-11 20:12:09 -06:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/stat.h>
|
2019-02-27 20:19:31 -06:00
|
|
|
#include <sys/ioctl.h>
|
2019-02-11 20:12:09 -06:00
|
|
|
#include <net/if.h>
|
|
|
|
|
|
|
|
#include <hexagram/capture.h>
|
|
|
|
|
|
|
|
struct _hexagram_capture {
|
|
|
|
int fd;
|
|
|
|
uint32_t endian;
|
|
|
|
};
|
|
|
|
|
|
|
|
hexagram_capture *hexagram_capture_open_fd(int fd, int flags) {
|
|
|
|
hexagram_capture *capture;
|
|
|
|
hexagram_capture_header header;
|
|
|
|
|
|
|
|
if ((capture = malloc(sizeof(*capture))) == NULL) {
|
|
|
|
goto error_malloc;
|
|
|
|
}
|
|
|
|
|
2019-02-12 00:00:13 -06:00
|
|
|
if ((flags & O_WRONLY) && !(flags & O_APPEND)) {
|
2019-02-11 20:12:09 -06:00
|
|
|
memcpy(header.magic, HEXAGRAM_CAPTURE_MAGIC, strlen(HEXAGRAM_CAPTURE_MAGIC));
|
|
|
|
|
|
|
|
header.endian = HEXAGRAM_CAPTURE_ENDIAN;
|
|
|
|
|
|
|
|
if (write(fd, &header, sizeof(header)) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
2019-02-12 00:00:13 -06:00
|
|
|
} else if (flags == O_RDONLY) {
|
2019-02-11 20:12:09 -06:00
|
|
|
ssize_t readlen;
|
|
|
|
|
|
|
|
if ((readlen = read(fd, &header, sizeof(header))) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memcmp(header.magic, HEXAGRAM_CAPTURE_MAGIC, sizeof(header.magic)) != 0) {
|
|
|
|
goto error_invalid_format;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (header.endian) {
|
|
|
|
case HEXAGRAM_CAPTURE_ENDIAN:
|
|
|
|
case HEXAGRAM_CAPTURE_ENDIAN_SWAPPED:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
goto error_invalid_format;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
capture->fd = fd;
|
|
|
|
capture->endian = header.endian;
|
|
|
|
|
|
|
|
return capture;
|
|
|
|
|
|
|
|
error_invalid_format:
|
|
|
|
error_io:
|
|
|
|
free(capture);
|
|
|
|
|
|
|
|
error_malloc:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
hexagram_capture *hexagram_capture_open_file(const char *file, int flags) {
|
|
|
|
int fd = (flags & O_CREAT)?
|
|
|
|
open(file, flags, 0644):
|
|
|
|
open(file, flags);
|
|
|
|
|
|
|
|
if (fd < 0) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return hexagram_capture_open_fd(fd, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hexagram_capture_destroy(hexagram_capture *capture) {
|
|
|
|
memset(capture, '\0', sizeof(*capture));
|
|
|
|
|
|
|
|
free(capture);
|
|
|
|
}
|
|
|
|
|
2019-02-12 00:00:13 -06:00
|
|
|
void hexagram_capture_close(hexagram_capture *capture) {
|
|
|
|
close(capture->fd);
|
|
|
|
|
|
|
|
hexagram_capture_destroy(capture);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t hexagram_capture_read(hexagram_capture *capture,
|
2019-02-19 19:26:33 -06:00
|
|
|
struct timeval *timestamp,
|
2019-02-12 00:00:13 -06:00
|
|
|
struct can_frame *frame) {
|
2019-02-11 20:12:09 -06:00
|
|
|
ssize_t len;
|
|
|
|
hexagram_capture_frame data;
|
|
|
|
|
|
|
|
if ((len = read(capture->fd, &data, sizeof(data))) < 0) {
|
|
|
|
goto error_io;
|
2019-02-16 01:39:43 -06:00
|
|
|
} else if (len && len < sizeof(data)) {
|
2019-02-11 20:12:09 -06:00
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
2019-02-19 19:26:33 -06:00
|
|
|
timestamp->tv_sec = data.sec;
|
|
|
|
timestamp->tv_usec = data.usec;
|
2019-02-11 20:12:09 -06:00
|
|
|
|
|
|
|
memcpy(frame, &data.frame, sizeof(data.frame));
|
|
|
|
|
2019-02-12 00:00:13 -06:00
|
|
|
return len;
|
2019-02-11 20:12:09 -06:00
|
|
|
|
|
|
|
error_io:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-02-12 00:00:13 -06:00
|
|
|
ssize_t hexagram_capture_write(hexagram_capture *capture,
|
2019-02-19 19:26:33 -06:00
|
|
|
struct timeval *timestamp,
|
|
|
|
struct can_frame *frame) {
|
2019-02-11 20:12:09 -06:00
|
|
|
hexagram_capture_frame data;
|
|
|
|
|
|
|
|
if (timestamp == NULL) {
|
|
|
|
struct timeval now;
|
|
|
|
|
|
|
|
if (gettimeofday(&now, NULL) < 0) {
|
|
|
|
goto error_gettimeofday;
|
|
|
|
}
|
|
|
|
|
2019-02-19 19:26:33 -06:00
|
|
|
data.sec = now.tv_sec;
|
|
|
|
data.usec = now.tv_usec;
|
2019-02-11 20:12:09 -06:00
|
|
|
} else {
|
2019-02-19 19:26:33 -06:00
|
|
|
data.sec = timestamp->tv_sec;
|
|
|
|
data.usec = timestamp->tv_usec;
|
2019-02-11 20:12:09 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(&data.frame, frame, sizeof(data.frame));
|
|
|
|
|
|
|
|
return write(capture->fd, &data, sizeof(data));
|
|
|
|
|
|
|
|
error_gettimeofday:
|
|
|
|
return -1;
|
|
|
|
}
|
2019-02-12 20:55:40 -06:00
|
|
|
|
2019-02-27 20:19:31 -06:00
|
|
|
int hexagram_capture_save(hexagram_capture *capture,
|
|
|
|
hexagram_can_if *can_if) {
|
|
|
|
struct can_frame frame;
|
|
|
|
|
2019-03-13 20:09:40 -05:00
|
|
|
while (hexagram_can_if_read(can_if, &frame) > 0) {
|
2019-02-27 20:19:31 -06:00
|
|
|
struct timeval timestamp;
|
|
|
|
|
|
|
|
if (ioctl(can_if->sock, SIOCGSTAMP, ×tamp) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hexagram_capture_write(capture, ×tamp, &frame) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error_io:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-02-12 20:55:40 -06:00
|
|
|
int hexagram_capture_replay(hexagram_capture *capture,
|
|
|
|
hexagram_can_if *can_if,
|
|
|
|
float speed) {
|
2019-02-19 19:26:33 -06:00
|
|
|
struct timeval timestamp;
|
|
|
|
|
|
|
|
uint64_t usec_last = 0,
|
|
|
|
usec;
|
2019-02-12 20:55:40 -06:00
|
|
|
|
|
|
|
struct can_frame frame;
|
|
|
|
|
|
|
|
while (1) {
|
2019-02-16 01:39:43 -06:00
|
|
|
ssize_t len = hexagram_capture_read(capture,
|
2019-02-19 19:26:33 -06:00
|
|
|
×tamp,
|
2019-02-16 01:39:43 -06:00
|
|
|
&frame);
|
|
|
|
|
|
|
|
if (len < 0) {
|
2019-02-12 20:55:40 -06:00
|
|
|
goto error_io;
|
|
|
|
} else if (len == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-02-19 19:26:33 -06:00
|
|
|
usec = timestamp.tv_sec * 1000000
|
|
|
|
+ timestamp.tv_usec;
|
|
|
|
|
2019-02-12 20:55:40 -06:00
|
|
|
if (usec_last) {
|
2019-02-19 19:26:33 -06:00
|
|
|
usleep(speed * (useconds_t)(usec - usec_last));
|
2019-02-12 20:55:40 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (write(can_if->sock, &frame, sizeof(struct can_frame)) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
2019-02-19 19:26:33 -06:00
|
|
|
usec_last = usec;
|
2019-02-12 20:55:40 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error_io:
|
|
|
|
return -1;
|
|
|
|
}
|