Commit 2f842304 authored by Bryan O'Donoghue's avatar Bryan O'Donoghue Committed by Greg Kroah-Hartman

greybus: loopback: add ability to graph greybus latency

This patch adds the ability to graph the latency of the overhead introduced
by the greybus stack in the roundtrip time AP->Module->AP.

Since this is a small number it is reported in nanoseconds, not
mircoseconds. This data can also be derived with tracepoints but it's being
provided in this format to export to CSV file more easily than with
tracepoints.
Signed-off-by: default avatarBryan O'Donoghue <bryan.odonoghue@linaro.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@google.com>
parent 464dc8cb
...@@ -50,9 +50,11 @@ struct gb_loopback { ...@@ -50,9 +50,11 @@ struct gb_loopback {
int ms_wait; int ms_wait;
struct gb_loopback_stats latency; struct gb_loopback_stats latency;
struct gb_loopback_stats latency_gb;
struct gb_loopback_stats throughput; struct gb_loopback_stats throughput;
struct gb_loopback_stats requests_per_second; struct gb_loopback_stats requests_per_second;
u64 elapsed_nsecs; u64 elapsed_nsecs;
u64 elapsed_nsecs_gb;
u32 error; u32 error;
}; };
...@@ -177,6 +179,8 @@ static void gb_loopback_check_attr(struct gb_connection *connection, ...@@ -177,6 +179,8 @@ static void gb_loopback_check_attr(struct gb_connection *connection,
/* Time to send and receive one message */ /* Time to send and receive one message */
gb_loopback_stats_attrs(latency); gb_loopback_stats_attrs(latency);
/* Time to send and receive one message not including greybus */
gb_loopback_stats_attrs(latency_gb);
/* Number of requests sent per second on this cport */ /* Number of requests sent per second on this cport */
gb_loopback_stats_attrs(requests_per_second); gb_loopback_stats_attrs(requests_per_second);
/* Quantity of data sent and received on this cport */ /* Quantity of data sent and received on this cport */
...@@ -209,6 +213,7 @@ gb_loopback_attr(iteration_max, u); ...@@ -209,6 +213,7 @@ gb_loopback_attr(iteration_max, u);
static struct attribute *loopback_attrs[] = { static struct attribute *loopback_attrs[] = {
dev_stats_attrs(latency), dev_stats_attrs(latency),
dev_stats_attrs(latency_gb),
dev_stats_attrs(requests_per_second), dev_stats_attrs(requests_per_second),
dev_stats_attrs(throughput), dev_stats_attrs(throughput),
&dev_attr_type.attr, &dev_attr_type.attr,
...@@ -221,17 +226,16 @@ static struct attribute *loopback_attrs[] = { ...@@ -221,17 +226,16 @@ static struct attribute *loopback_attrs[] = {
}; };
ATTRIBUTE_GROUPS(loopback); ATTRIBUTE_GROUPS(loopback);
static void gb_loopback_calc_latency(struct gb_loopback *gb, static u64 gb_loopback_calc_latency(struct timeval *ts, struct timeval *te)
struct timeval *ts, struct timeval *te)
{ {
u64 t1, t2; u64 t1, t2;
t1 = timeval_to_ns(ts); t1 = timeval_to_ns(ts);
t2 = timeval_to_ns(te); t2 = timeval_to_ns(te);
if (t2 > t1) if (t2 > t1)
gb->elapsed_nsecs = t2 - t1; return t2 - t1;
else else
gb->elapsed_nsecs = NSEC_PER_DAY - t2 + t1; return NSEC_PER_DAY - t2 + t1;
} }
static int gb_loopback_sink(struct gb_loopback *gb, u32 len) static int gb_loopback_sink(struct gb_loopback *gb, u32 len)
...@@ -251,7 +255,15 @@ static int gb_loopback_sink(struct gb_loopback *gb, u32 len) ...@@ -251,7 +255,15 @@ static int gb_loopback_sink(struct gb_loopback *gb, u32 len)
request, len + sizeof(*request), NULL, 0); request, len + sizeof(*request), NULL, 0);
do_gettimeofday(&te); do_gettimeofday(&te);
gb_loopback_calc_latency(gb, &ts, &te);
/* Calculate the total time the message took */
gb->elapsed_nsecs = gb_loopback_calc_latency(&ts, &te);
/* Calculate non-greybus related component of the latency */
gb_connection_pop_timestamp(gb->connection, &ts);
gb_connection_pop_timestamp(gb->connection, &te);
gb->elapsed_nsecs_gb = gb_loopback_calc_latency(&ts, &te);
kfree(request); kfree(request);
return retval; return retval;
...@@ -282,7 +294,14 @@ static int gb_loopback_transfer(struct gb_loopback *gb, u32 len) ...@@ -282,7 +294,14 @@ static int gb_loopback_transfer(struct gb_loopback *gb, u32 len)
request, len + sizeof(*request), request, len + sizeof(*request),
response, len + sizeof(*response)); response, len + sizeof(*response));
do_gettimeofday(&te); do_gettimeofday(&te);
gb_loopback_calc_latency(gb, &ts, &te);
/* Calculate the total time the message took */
gb->elapsed_nsecs = gb_loopback_calc_latency(&ts, &te);
/* Calculate non-greybus related component of the latency */
gb_connection_pop_timestamp(gb->connection, &ts);
gb_connection_pop_timestamp(gb->connection, &te);
gb->elapsed_nsecs_gb = gb_loopback_calc_latency(&ts, &te);
if (retval) if (retval)
goto gb_error; goto gb_error;
...@@ -308,7 +327,14 @@ static int gb_loopback_ping(struct gb_loopback *gb) ...@@ -308,7 +327,14 @@ static int gb_loopback_ping(struct gb_loopback *gb)
retval = gb_operation_sync(gb->connection, GB_LOOPBACK_TYPE_PING, retval = gb_operation_sync(gb->connection, GB_LOOPBACK_TYPE_PING,
NULL, 0, NULL, 0); NULL, 0, NULL, 0);
do_gettimeofday(&te); do_gettimeofday(&te);
gb_loopback_calc_latency(gb, &ts, &te);
/* Calculate the total time the message took */
gb->elapsed_nsecs = gb_loopback_calc_latency(&ts, &te);
/* Calculate non-greybus related component of the latency */
gb_connection_pop_timestamp(gb->connection, &ts);
gb_connection_pop_timestamp(gb->connection, &te);
gb->elapsed_nsecs_gb = gb_loopback_calc_latency(&ts, &te);
return retval; return retval;
} }
...@@ -371,6 +397,7 @@ static void gb_loopback_reset_stats(struct gb_loopback *gb) ...@@ -371,6 +397,7 @@ static void gb_loopback_reset_stats(struct gb_loopback *gb)
.min = U32_MAX, .min = U32_MAX,
}; };
memcpy(&gb->latency, &reset, sizeof(struct gb_loopback_stats)); memcpy(&gb->latency, &reset, sizeof(struct gb_loopback_stats));
memcpy(&gb->latency_gb, &reset, sizeof(struct gb_loopback_stats));
memcpy(&gb->throughput, &reset, sizeof(struct gb_loopback_stats)); memcpy(&gb->throughput, &reset, sizeof(struct gb_loopback_stats));
memcpy(&gb->requests_per_second, &reset, memcpy(&gb->requests_per_second, &reset,
sizeof(struct gb_loopback_stats)); sizeof(struct gb_loopback_stats));
...@@ -439,6 +466,11 @@ static void gb_loopback_calculate_stats(struct gb_loopback *gb) ...@@ -439,6 +466,11 @@ static void gb_loopback_calculate_stats(struct gb_loopback *gb)
/* Log throughput and requests using latency as benchmark */ /* Log throughput and requests using latency as benchmark */
gb_loopback_throughput_update(gb, lat); gb_loopback_throughput_update(gb, lat);
gb_loopback_requests_update(gb, lat); gb_loopback_requests_update(gb, lat);
/* Calculate the greybus related latency number in nanoseconds */
tmp = gb->elapsed_nsecs - gb->elapsed_nsecs_gb;
lat = tmp;
gb_loopback_update_stats(&gb->latency_gb, lat);
} }
static int gb_loopback_fn(void *data) static int gb_loopback_fn(void *data)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment