Commit 7c0ee8fd authored by Mika Westerberg's avatar Mika Westerberg

thunderbolt: Report consumed bandwidth in both directions

Whereas DisplayPort bandwidth is consumed only in one direction (from DP
IN adapter to DP OUT adapter), USB3 adds separate bandwidth for both
upstream and downstream directions.

For this reason extend the tunnel consumed bandwidth routines to support
both directions and implement this for DP.
Signed-off-by: default avatarMika Westerberg <mika.westerberg@linux.intel.com>
parent 9cac51a0
...@@ -535,7 +535,7 @@ static int tb_available_bw(struct tb_cm *tcm, struct tb_port *in, ...@@ -535,7 +535,7 @@ static int tb_available_bw(struct tb_cm *tcm, struct tb_port *in,
{ {
struct tb_switch *sw = out->sw; struct tb_switch *sw = out->sw;
struct tb_tunnel *tunnel; struct tb_tunnel *tunnel;
int bw, available_bw = 40000; int ret, bw, available_bw = 40000;
while (sw && sw != in->sw) { while (sw && sw != in->sw) {
bw = sw->link_speed * sw->link_width * 1000; /* Mb/s */ bw = sw->link_speed * sw->link_width * 1000; /* Mb/s */
...@@ -553,9 +553,10 @@ static int tb_available_bw(struct tb_cm *tcm, struct tb_port *in, ...@@ -553,9 +553,10 @@ static int tb_available_bw(struct tb_cm *tcm, struct tb_port *in,
if (!tb_tunnel_switch_on_path(tunnel, sw)) if (!tb_tunnel_switch_on_path(tunnel, sw))
continue; continue;
consumed_bw = tb_tunnel_consumed_bandwidth(tunnel); ret = tb_tunnel_consumed_bandwidth(tunnel, NULL,
if (consumed_bw < 0) &consumed_bw);
return consumed_bw; if (ret)
return ret;
bw -= consumed_bw; bw -= consumed_bw;
} }
......
...@@ -536,7 +536,8 @@ static int tb_dp_activate(struct tb_tunnel *tunnel, bool active) ...@@ -536,7 +536,8 @@ static int tb_dp_activate(struct tb_tunnel *tunnel, bool active)
return 0; return 0;
} }
static int tb_dp_consumed_bandwidth(struct tb_tunnel *tunnel) static int tb_dp_consumed_bandwidth(struct tb_tunnel *tunnel, int *consumed_up,
int *consumed_down)
{ {
struct tb_port *in = tunnel->src_port; struct tb_port *in = tunnel->src_port;
const struct tb_switch *sw = in->sw; const struct tb_switch *sw = in->sw;
...@@ -580,10 +581,20 @@ static int tb_dp_consumed_bandwidth(struct tb_tunnel *tunnel) ...@@ -580,10 +581,20 @@ static int tb_dp_consumed_bandwidth(struct tb_tunnel *tunnel)
lanes = tb_dp_cap_get_lanes(val); lanes = tb_dp_cap_get_lanes(val);
} else { } else {
/* No bandwidth management for legacy devices */ /* No bandwidth management for legacy devices */
*consumed_up = 0;
*consumed_down = 0;
return 0; return 0;
} }
return tb_dp_bandwidth(rate, lanes); if (in->sw->config.depth < tunnel->dst_port->sw->config.depth) {
*consumed_up = 0;
*consumed_down = tb_dp_bandwidth(rate, lanes);
} else {
*consumed_up = tb_dp_bandwidth(rate, lanes);
*consumed_down = 0;
}
return 0;
} }
static void tb_dp_init_aux_path(struct tb_path *path) static void tb_dp_init_aux_path(struct tb_path *path)
...@@ -1174,21 +1185,39 @@ static bool tb_tunnel_is_active(const struct tb_tunnel *tunnel) ...@@ -1174,21 +1185,39 @@ static bool tb_tunnel_is_active(const struct tb_tunnel *tunnel)
/** /**
* tb_tunnel_consumed_bandwidth() - Return bandwidth consumed by the tunnel * tb_tunnel_consumed_bandwidth() - Return bandwidth consumed by the tunnel
* @tunnel: Tunnel to check * @tunnel: Tunnel to check
* @consumed_up: Consumed bandwidth in Mb/s from @dst_port to @src_port.
* Can be %NULL.
* @consumed_down: Consumed bandwidth in Mb/s from @src_port to @dst_port.
* Can be %NULL.
* *
* Returns bandwidth currently consumed by @tunnel and %0 if the @tunnel * Stores the amount of isochronous bandwidth @tunnel consumes in
* is not active or does consume bandwidth. * @consumed_up and @consumed_down. In case of success returns %0,
* negative errno otherwise.
*/ */
int tb_tunnel_consumed_bandwidth(struct tb_tunnel *tunnel) int tb_tunnel_consumed_bandwidth(struct tb_tunnel *tunnel, int *consumed_up,
int *consumed_down)
{ {
int up_bw = 0, down_bw = 0;
if (!tb_tunnel_is_active(tunnel)) if (!tb_tunnel_is_active(tunnel))
return 0; goto out;
if (tunnel->consumed_bandwidth) { if (tunnel->consumed_bandwidth) {
int ret = tunnel->consumed_bandwidth(tunnel); int ret;
tb_tunnel_dbg(tunnel, "consumed bandwidth %d Mb/s\n", ret); ret = tunnel->consumed_bandwidth(tunnel, &up_bw, &down_bw);
if (ret)
return ret; return ret;
tb_tunnel_dbg(tunnel, "consumed bandwidth %d/%d Mb/s\n", up_bw,
down_bw);
} }
out:
if (consumed_up)
*consumed_up = up_bw;
if (consumed_down)
*consumed_down = down_bw;
return 0; return 0;
} }
...@@ -42,7 +42,8 @@ struct tb_tunnel { ...@@ -42,7 +42,8 @@ struct tb_tunnel {
size_t npaths; size_t npaths;
int (*init)(struct tb_tunnel *tunnel); int (*init)(struct tb_tunnel *tunnel);
int (*activate)(struct tb_tunnel *tunnel, bool activate); int (*activate)(struct tb_tunnel *tunnel, bool activate);
int (*consumed_bandwidth)(struct tb_tunnel *tunnel); int (*consumed_bandwidth)(struct tb_tunnel *tunnel, int *consumed_up,
int *consumed_down);
struct list_head list; struct list_head list;
enum tb_tunnel_type type; enum tb_tunnel_type type;
unsigned int max_bw; unsigned int max_bw;
...@@ -69,7 +70,8 @@ void tb_tunnel_deactivate(struct tb_tunnel *tunnel); ...@@ -69,7 +70,8 @@ void tb_tunnel_deactivate(struct tb_tunnel *tunnel);
bool tb_tunnel_is_invalid(struct tb_tunnel *tunnel); bool tb_tunnel_is_invalid(struct tb_tunnel *tunnel);
bool tb_tunnel_switch_on_path(const struct tb_tunnel *tunnel, bool tb_tunnel_switch_on_path(const struct tb_tunnel *tunnel,
const struct tb_switch *sw); const struct tb_switch *sw);
int tb_tunnel_consumed_bandwidth(struct tb_tunnel *tunnel); int tb_tunnel_consumed_bandwidth(struct tb_tunnel *tunnel, int *consumed_up,
int *consumed_down);
static inline bool tb_tunnel_is_pci(const struct tb_tunnel *tunnel) static inline bool tb_tunnel_is_pci(const struct tb_tunnel *tunnel)
{ {
......
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