.length = 2,
};
+struct {
+ unsigned enobufs;
+ unsigned overrun;
+ unsigned lost;
+} stats;
+
int num_interfaces = 0;
int count = 0; /* Number of sent messages */
+unsigned msg_in_progress = 0;
int completion_pipe[2];
struct msg_info {
if (idx >= MAX_INFOS)
error(1, 0, "%s idx too high", __FUNCTION__);
} else {
-
+
error(1, 0, "%s error", __FUNCTION__);
}
return &msg_infos[idx];
char *debugfs;
char path[256];
FILE *f;
-
+
debugfs = "/sys/kernel/debug";
if (debugfs) {
strcpy(path, debugfs);
mi->length = frame.can_dlc;
get_tstamp(&mi->ts_sent);
mi->sent = frame;
-
+
trace_on();
ret = write(socket, &frame, sizeof(frame));
trace_off(ret);
return ret;
}
+static inline send_and_check(int s)
+{
+ int ret;
+ ret = send_frame(s);
+ if (ret != sizeof(struct can_frame)) {
+/* if (ret == -1 && errno == ENOBUFS && opt.period_us == 0 && !opt.oneattime) { */
+/* stats.enobufs++; */
+/* /\* Ignore this error - pfifo_fast qeuue is full *\/ */
+/* } else */
+ error(1, errno, "send_frame (line %d)", __LINE__);
+ } else {
+ count++;
+ msg_in_progress++;
+ }
+}
+
static inline void get_next_timeout(struct timespec *timeout)
{
struct timespec now;
last.tv_nsec -= 1000000000;
last.tv_sec++;
}
- timespec_subtract(timeout, &last, &now);
+ if (timespec_subtract(timeout, &last, &now) /* is negative */) {
+ stats.overrun++;
+ memset(timeout, 0, sizeof(*timeout));
+ }
} else if (opt.timeout_ms != 0) {
timeout->tv_sec = opt.timeout_ms/1000;
timeout->tv_nsec = (opt.timeout_ms%1000)*1000000;
struct can_frame frame;
struct msg_info *mi;
int ret;
-
+
receive(s, &frame, &ts_kern, &ts_user);
mi = frame2info(&frame);
mi->ts_rx_final_kern = ts_kern;
struct timespec timeout;
struct sockaddr_can addr;
sigset_t set;
- unsigned msg_in_progress = 0;
MEMSET_ZERO(pfd);
set_sched_policy_and_prio(SCHED_FIFO, 40);
-#define SEND() \
- do { \
- ret = send_frame(pfd[0].fd); \
- if (ret != sizeof(struct can_frame)) { \
- if (ret == -1 && errno == ENOBUFS && opt.period_us == 0 && !opt.oneattime) { \
- /* Ignore this error - pfifo_fast qeuue is full */ \
- } else \
- error(1, errno, "send_frame (line %d)", __LINE__); \
- } \
- else { \
- count++; \
- msg_in_progress++; \
- } \
- } while (0)
-
- if (opt.oneattime) {
+#define SEND() send_and_check(pfd[0].fd)
+
+ if (opt.oneattime)
SEND();
- count = 1;
- }
while (!finish_flag &&
(opt.count == 0 || count < opt.count || msg_in_progress != 0)) {
return 0;
}
+void print_progress()
+{
+ printf("\rSent %5d, in progress %5d", count, msg_in_progress);
+ fflush(stdout);
+}
int main(int argc, const char *argv[])
{
error(1, errno, "pipe fcntl");
init_ftrace();
-
+
pthread_create(&thread, 0, measure_thread, NULL);
- struct timespec next, now, diff;
+ struct timespec next, now, diff, allsent = {0,0};
clock_gettime(CLOCK_MONOTONIC, &next);
int completed = 0;
while (!finish_flag && (opt.count == 0 || completed < opt.count)) {
clock_gettime(CLOCK_MONOTONIC, &now);
if (timespec_subtract(&diff, &next, &now)) {
- printf("\rMessage %d", count);
- fflush(stdout);
- next.tv_nsec += 100000000;
+ print_progress();
+ next.tv_nsec += 100000000;
while (next.tv_nsec >= 1000000000) {
next.tv_nsec -= 1000000000;
next.tv_sec++;
}
}
+ if (opt.count != 0 && count >= opt.count) {
+ if (allsent.tv_sec == 0)
+ allsent = now;
+ timespec_subtract(&diff, &now, &allsent);
+ if (diff.tv_sec >= 1)
+ finish_flag = 1;
+ }
}
- printf("\rMessage %d\n", count);
+ print_progress();
+ printf("\n");
+
+ stats.lost = msg_in_progress;
pthread_join(thread, NULL);
if (opt.file)
fclose(opt.file);
+ if (stats.overrun)
+ printf("overrun=%d\n", stats.overrun);
+ if (stats.enobufs)
+ printf("enobufs=%d\n", stats.enobufs);
+ if (stats.lost)
+ printf("lost=%d\n", stats.lost);
+
return 0;
}