|
@@ -21,6 +21,7 @@
|
|
|
#include <net/netlink.h>
|
|
|
#include <net/sch_generic.h>
|
|
|
#include <net/pkt_sched.h>
|
|
|
+#include <net/tcp.h>
|
|
|
|
|
|
|
|
|
/* Simple Token Bucket Filter.
|
|
@@ -117,6 +118,22 @@ struct tbf_sched_data {
|
|
|
};
|
|
|
|
|
|
|
|
|
+/*
|
|
|
+ * Return length of individual segments of a gso packet,
|
|
|
+ * including all headers (MAC, IP, TCP/UDP)
|
|
|
+ */
|
|
|
+static unsigned int skb_gso_seglen(const struct sk_buff *skb)
|
|
|
+{
|
|
|
+ unsigned int hdr_len = skb_transport_header(skb) - skb_mac_header(skb);
|
|
|
+ const struct skb_shared_info *shinfo = skb_shinfo(skb);
|
|
|
+
|
|
|
+ if (likely(shinfo->gso_type & (SKB_GSO_TCPV4 | SKB_GSO_TCPV6)))
|
|
|
+ hdr_len += tcp_hdrlen(skb);
|
|
|
+ else
|
|
|
+ hdr_len += sizeof(struct udphdr);
|
|
|
+ return hdr_len + shinfo->gso_size;
|
|
|
+}
|
|
|
+
|
|
|
/* GSO packet is too big, segment it so that tbf can transmit
|
|
|
* each segment in time
|
|
|
*/
|
|
@@ -136,12 +153,8 @@ static int tbf_segment(struct sk_buff *skb, struct Qdisc *sch)
|
|
|
while (segs) {
|
|
|
nskb = segs->next;
|
|
|
segs->next = NULL;
|
|
|
- if (likely(segs->len <= q->max_size)) {
|
|
|
- qdisc_skb_cb(segs)->pkt_len = segs->len;
|
|
|
- ret = qdisc_enqueue(segs, q->qdisc);
|
|
|
- } else {
|
|
|
- ret = qdisc_reshape_fail(skb, sch);
|
|
|
- }
|
|
|
+ qdisc_skb_cb(segs)->pkt_len = segs->len;
|
|
|
+ ret = qdisc_enqueue(segs, q->qdisc);
|
|
|
if (ret != NET_XMIT_SUCCESS) {
|
|
|
if (net_xmit_drop_count(ret))
|
|
|
sch->qstats.drops++;
|
|
@@ -163,7 +176,7 @@ static int tbf_enqueue(struct sk_buff *skb, struct Qdisc *sch)
|
|
|
int ret;
|
|
|
|
|
|
if (qdisc_pkt_len(skb) > q->max_size) {
|
|
|
- if (skb_is_gso(skb))
|
|
|
+ if (skb_is_gso(skb) && skb_gso_seglen(skb) <= q->max_size)
|
|
|
return tbf_segment(skb, sch);
|
|
|
return qdisc_reshape_fail(skb, sch);
|
|
|
}
|
|
@@ -319,6 +332,11 @@ static int tbf_change(struct Qdisc *sch, struct nlattr *opt)
|
|
|
if (max_size < 0)
|
|
|
goto done;
|
|
|
|
|
|
+ if (max_size < psched_mtu(qdisc_dev(sch)))
|
|
|
+ pr_warn_ratelimited("sch_tbf: burst %u is lower than device %s mtu (%u) !\n",
|
|
|
+ max_size, qdisc_dev(sch)->name,
|
|
|
+ psched_mtu(qdisc_dev(sch)));
|
|
|
+
|
|
|
if (q->qdisc != &noop_qdisc) {
|
|
|
err = fifo_set_limit(q->qdisc, qopt->limit);
|
|
|
if (err)
|