* [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint
@ 2026-02-06 16:24 Jesper Dangaard Brouer
2026-02-06 16:24 ` [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing Jesper Dangaard Brouer
` (5 more replies)
0 siblings, 6 replies; 17+ messages in thread
From: Jesper Dangaard Brouer @ 2026-02-06 16:24 UTC (permalink / raw)
To: netdev, Eric Dumazet, David S. Miller, Paolo Abeni,
Toke Høiland-Jørgensen
Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri,
edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team
This series refactors qdisc drop reason handling by introducing a dedicated
enum qdisc_drop_reason and trace_qdisc_drop tracepoint, providing qdisc
layer drop diagnostics with direct qdisc context visibility.
Background:
-----------
Identifying which qdisc dropped a packet via skb_drop_reason is difficult.
Normally, the kfree_skb tracepoint caller "location" hints at the dropping
code, but qdisc drops happen at a central point (__dev_queue_xmit), making
this unusable. As a workaround, commits 5765c7f6e317 ("net_sched: sch_fq:
add three drop_reason") and a42d71e322a8 ("net_sched: sch_cake: Add drop
reasons") encoded qdisc names directly in the drop reason enums.
This series provides a cleaner solution by creating a dedicated qdisc
tracepoint that naturally includes qdisc context (handle, parent, kind).
Solution:
---------
Create a new tracepoint trace_qdisc_drop that builds on top of existing
trace_qdisc_enqueue infrastructure. It includes qdisc handle, parent,
qdisc kind (name), and device information directly.
The existing SKB_DROP_REASON_QDISC_DROP is retained for backwards
compatibility via kfree_skb_reason(). The qdisc-specific drop reasons
(QDISC_DROP_*) provide fine-grained detail via the new tracepoint.
The enum uses subsystem encoding (offset by SKB_DROP_REASON_SUBSYS_QDISC)
to catch type mismatches during debugging.
This implements the alternative approach described in:
https://lore.kernel.org/all/6be17a08-f8aa-4f91-9bd0-d9e1f0a92d90@kernel.org/
Changes since v2:
-----------------
Link v2: https://lore.kernel.org/all/177032644012.1975497.16411100029657607833.stgit@firesoul/
- Patch 1: Fix tcf_kfree_skb_list() to handle TC classifier drop reasons
- syzbot triggered WARN_ON_ONCE revealing TC classifier's SKB_DROP_REASON_TC_*
reasons arrive via shared tc_skb_cb->drop_reason storage
- Use SKB_DROP_REASON_SUBSYS_MASK to distinguish qdisc vs TC classifier reasons
- Pass TC classifier reasons through to kfree_skb_reason() unchanged
- Reported-by: syzbot+ci335f183b8446659a@syzkaller.appspotmail.com
- Patch 6: Fix compile error - add missing QDISC_DROP_L4S_STEP_NON_ECN
- Reported-by: kernel test robot <lkp@intel.com>
- Closes: https://lore.kernel.org/oe-kbuild-all/202602061240.CRDtcOeX-lkp@intel.com/
- Reported-by: kernel test robot <lkp@intel.com>
Changes since RFC v1:
---------------------
Link RFC-v1: https://lore.kernel.org/all/177022452988.1827734.5121740962985640333.stgit@firesoul/
- Added SKB_DROP_REASON_SUBSYS_QDISC subsystem encoding for enum debugging
- Added WARN_ON_ONCE to catch wrong enum type (skb vs qdisc drop reason)
- Renamed QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION (Toke)
- Renamed QDISC_DROP_FQ_BAND_LIMIT to QDISC_DROP_BAND_LIMIT
- Renamed QDISC_DROP_FQ_HORIZON_LIMIT to QDISC_DROP_HORIZON_LIMIT
- Updated SKB_DROP_REASON_QDISC_DROP doc to reference qdisc:qdisc_drop
- Split rename changes into separate patches for clarity
Series overview:
----------------
Patch 1: Introduce the qdisc drop reason infrastructure
- New enum qdisc_drop_reason with subsystem encoding
- New trace_qdisc_drop tracepoint with qdisc context
- tcf_kfree_skb_list() handles both qdisc and TC classifier drop reasons
- Convert FQ, FQ_CoDel, CoDel, SFB, pfifo_fast to use new infrastructure
Patch 2: Convert SFQ as example for reviewers
- Demonstrates how to convert a flow-based qdisc
- Adds QDISC_DROP_MAXFLOWS for flow table exhaustion
Patch 3: Fix CAKE to use enum qdisc_drop_reason
- Change cobalt_should_drop() return type for type correctness
Patch 4: Rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION
- Generic name without embedding qdisc name
Patch 5: Rename QDISC_DROP_FQ_* to generic names
- QDISC_DROP_FQ_BAND_LIMIT -> QDISC_DROP_BAND_LIMIT
- QDISC_DROP_FQ_HORIZON_LIMIT -> QDISC_DROP_HORIZON_LIMIT
Patch 6: DualPI2 use qdisc_dequeue_drop() for dequeue drops
- Demonstrates dequeue-time drop handling with trace_qdisc_drop
- Adds QDISC_DROP_L4S_STEP_NON_ECN for L4S step AQM drops
---
Jesper Dangaard Brouer (6):
net: sched: introduce qdisc-specific drop reason tracing
net: sched: sfq: convert to qdisc drop reasons
net: sched: sch_cake: use enum qdisc_drop_reason for cobalt_should_drop
net: sched: rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION
net: sched: rename QDISC_DROP_FQ_* to generic names
net: sched: sch_dualpi2: use qdisc_dequeue_drop() for dequeue drops
include/net/dropreason-core.h | 48 ++------------
include/net/dropreason-qdisc.h | 115 +++++++++++++++++++++++++++++++++
include/net/dropreason.h | 6 ++
include/net/sch_generic.h | 36 +++++++----
include/trace/events/qdisc.h | 51 +++++++++++++++
net/core/dev.c | 8 +--
net/sched/sch_cake.c | 26 ++++----
net/sched/sch_codel.c | 5 +-
net/sched/sch_dualpi2.c | 18 +++---
net/sched/sch_fq.c | 7 +-
net/sched/sch_fq_codel.c | 4 +-
net/sched/sch_fq_pie.c | 4 +-
net/sched/sch_generic.c | 29 ++++++++-
net/sched/sch_gred.c | 4 +-
net/sched/sch_pie.c | 4 +-
net/sched/sch_red.c | 4 +-
net/sched/sch_sfb.c | 4 +-
net/sched/sch_sfq.c | 8 +--
18 files changed, 272 insertions(+), 109 deletions(-)
create mode 100644 include/net/dropreason-qdisc.h
--
^ permalink raw reply [flat|nested] 17+ messages in thread* [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing 2026-02-06 16:24 [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint Jesper Dangaard Brouer @ 2026-02-06 16:24 ` Jesper Dangaard Brouer 2026-02-10 11:47 ` Toke Høiland-Jørgensen ` (2 more replies) 2026-02-06 16:24 ` [PATCH net-next v3 2/6] net: sched: sfq: convert to qdisc drop reasons Jesper Dangaard Brouer ` (4 subsequent siblings) 5 siblings, 3 replies; 17+ messages in thread From: Jesper Dangaard Brouer @ 2026-02-06 16:24 UTC (permalink / raw) To: netdev, Eric Dumazet, David S. Miller, Paolo Abeni, Toke Høiland-Jørgensen Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Create new enum qdisc_drop_reason and trace_qdisc_drop tracepoint for qdisc layer drop diagnostics with direct qdisc context visibility. The new tracepoint includes qdisc handle, parent, kind (name), and device information. Existing SKB_DROP_REASON_QDISC_DROP is retained for backwards compatibility via kfree_skb_reason(). Convert FQ, FQ_CoDel, CoDel, SFB, and pfifo_fast to use the new infrastructure. Reported-by: syzbot+ci335f183b8446659a@syzkaller.appspotmail.com Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> --- include/net/dropreason-core.h | 42 ++---------------- include/net/dropreason-qdisc.h | 95 ++++++++++++++++++++++++++++++++++++++++ include/net/dropreason.h | 6 +++ include/net/sch_generic.h | 36 ++++++++++----- include/trace/events/qdisc.h | 51 +++++++++++++++++++++ net/core/dev.c | 8 ++- net/sched/sch_cake.c | 6 +-- net/sched/sch_codel.c | 5 +- net/sched/sch_dualpi2.c | 8 +-- net/sched/sch_fq.c | 7 +-- net/sched/sch_fq_codel.c | 4 +- net/sched/sch_fq_pie.c | 4 +- net/sched/sch_generic.c | 29 +++++++++++- net/sched/sch_gred.c | 4 +- net/sched/sch_pie.c | 4 +- net/sched/sch_red.c | 4 +- net/sched/sch_sfb.c | 4 +- 17 files changed, 233 insertions(+), 84 deletions(-) create mode 100644 include/net/dropreason-qdisc.h diff --git a/include/net/dropreason-core.h b/include/net/dropreason-core.h index a7b7abd66e21..3d8d284e05c8 100644 --- a/include/net/dropreason-core.h +++ b/include/net/dropreason-core.h @@ -68,12 +68,6 @@ FN(SECURITY_HOOK) \ FN(QDISC_DROP) \ FN(QDISC_BURST_DROP) \ - FN(QDISC_OVERLIMIT) \ - FN(QDISC_CONGESTED) \ - FN(CAKE_FLOOD) \ - FN(FQ_BAND_LIMIT) \ - FN(FQ_HORIZON_LIMIT) \ - FN(FQ_FLOW_LIMIT) \ FN(CPU_BACKLOG) \ FN(XDP) \ FN(TC_INGRESS) \ @@ -371,8 +365,10 @@ enum skb_drop_reason { /** @SKB_DROP_REASON_SECURITY_HOOK: dropped due to security HOOK */ SKB_DROP_REASON_SECURITY_HOOK, /** - * @SKB_DROP_REASON_QDISC_DROP: dropped by qdisc when packet outputting ( - * failed to enqueue to current qdisc) + * @SKB_DROP_REASON_QDISC_DROP: dropped by qdisc during enqueue or + * dequeue. More specific drop reasons are available via the + * qdisc:qdisc_drop tracepoint, which also provides qdisc handle + * and name for identifying the source. */ SKB_DROP_REASON_QDISC_DROP, /** @@ -380,36 +376,6 @@ enum skb_drop_reason { * limit is hit. */ SKB_DROP_REASON_QDISC_BURST_DROP, - /** - * @SKB_DROP_REASON_QDISC_OVERLIMIT: dropped by qdisc when a qdisc - * instance exceeds its total buffer size limit. - */ - SKB_DROP_REASON_QDISC_OVERLIMIT, - /** - * @SKB_DROP_REASON_QDISC_CONGESTED: dropped by a qdisc AQM algorithm - * due to congestion. - */ - SKB_DROP_REASON_QDISC_CONGESTED, - /** - * @SKB_DROP_REASON_CAKE_FLOOD: dropped by the flood protection part of - * CAKE qdisc AQM algorithm (BLUE). - */ - SKB_DROP_REASON_CAKE_FLOOD, - /** - * @SKB_DROP_REASON_FQ_BAND_LIMIT: dropped by fq qdisc when per band - * limit is reached. - */ - SKB_DROP_REASON_FQ_BAND_LIMIT, - /** - * @SKB_DROP_REASON_FQ_HORIZON_LIMIT: dropped by fq qdisc when packet - * timestamp is too far in the future. - */ - SKB_DROP_REASON_FQ_HORIZON_LIMIT, - /** - * @SKB_DROP_REASON_FQ_FLOW_LIMIT: dropped by fq qdisc when a flow - * exceeds its limits. - */ - SKB_DROP_REASON_FQ_FLOW_LIMIT, /** * @SKB_DROP_REASON_CPU_BACKLOG: failed to enqueue the skb to the per CPU * backlog queue. This can be caused by backlog queue full (see diff --git a/include/net/dropreason-qdisc.h b/include/net/dropreason-qdisc.h new file mode 100644 index 000000000000..db81b3471a65 --- /dev/null +++ b/include/net/dropreason-qdisc.h @@ -0,0 +1,95 @@ +/* SPDX-License-Identifier: GPL-2.0-or-later */ + +#ifndef _LINUX_DROPREASON_QDISC_H +#define _LINUX_DROPREASON_QDISC_H +#include <net/dropreason.h> + +#define DEFINE_QDISC_DROP_REASON(FN, FNe) \ + FN(UNSPEC) \ + FN(GENERIC) \ + FN(OVERLIMIT) \ + FN(CONGESTED) \ + FN(CAKE_FLOOD) \ + FN(FQ_BAND_LIMIT) \ + FN(FQ_HORIZON_LIMIT) \ + FN(FQ_FLOW_LIMIT) \ + FNe(MAX) + +#undef FN +#undef FNe +#define FN(reason) QDISC_DROP_##reason, +#define FNe(reason) QDISC_DROP_##reason + +/** + * enum qdisc_drop_reason - reason why a qdisc dropped a packet + * + * Qdisc-specific drop reasons for packet drops that occur within the + * traffic control (TC) queueing discipline layer. These reasons provide + * detailed diagnostics about why packets were dropped by various qdisc + * algorithms, enabling fine-grained monitoring and troubleshooting of + * queue behavior. + */ +enum qdisc_drop_reason { + /** + * @QDISC_DROP_UNSPEC: unspecified/invalid qdisc drop reason. + * Value 0 serves as analogous to SKB_NOT_DROPPED_YET for enum skb_drop_reason. + * Used for catching zero-initialized drop_reason fields. + */ + QDISC_DROP_UNSPEC = 0, + /** + * @__QDISC_DROP_REASON: subsystem base value for qdisc drop reasons + */ + __QDISC_DROP_REASON = SKB_DROP_REASON_SUBSYS_QDISC << + SKB_DROP_REASON_SUBSYS_SHIFT, + /** + * @QDISC_DROP_GENERIC: generic/default qdisc drop, used when no + * more specific reason applies + */ + QDISC_DROP_GENERIC, + /** + * @QDISC_DROP_OVERLIMIT: packet dropped because the qdisc queue + * length exceeded its configured limit (sch->limit). This typically + * indicates the queue is full and cannot accept more packets. + */ + QDISC_DROP_OVERLIMIT, + /** + * @QDISC_DROP_CONGESTED: packet dropped due to active congestion + * control algorithms (e.g., CoDel, PIE, RED) detecting network + * congestion. The qdisc proactively dropped the packet to signal + * congestion to the sender and prevent bufferbloat. + */ + QDISC_DROP_CONGESTED, + /** + * @QDISC_DROP_CAKE_FLOOD: CAKE qdisc dropped packet due to flood + * protection mechanism (BLUE algorithm). This indicates potential + * DoS/flood attack or unresponsive flow behavior. + */ + QDISC_DROP_CAKE_FLOOD, + /** + * @QDISC_DROP_FQ_BAND_LIMIT: FQ (Fair Queue) dropped packet because + * the priority band's packet limit was reached. Each priority band + * in FQ has its own limit. + */ + QDISC_DROP_FQ_BAND_LIMIT, + /** + * @QDISC_DROP_FQ_HORIZON_LIMIT: FQ dropped packet because its + * timestamp is too far in the future (beyond horizon). This prevents + * packets with unreasonable future timestamps from blocking the queue. + */ + QDISC_DROP_FQ_HORIZON_LIMIT, + /** + * @QDISC_DROP_FQ_FLOW_LIMIT: FQ dropped packet because an individual + * flow exceeded its per-flow packet limit. + */ + QDISC_DROP_FQ_FLOW_LIMIT, + /** + * @QDISC_DROP_MAX: the maximum of qdisc drop reasons, which + * shouldn't be used as a real 'reason' - only for tracing code gen + */ + QDISC_DROP_MAX, +}; + +#undef FN +#undef FNe + +#endif diff --git a/include/net/dropreason.h b/include/net/dropreason.h index 7d3b1a2a6fec..1df60645fb27 100644 --- a/include/net/dropreason.h +++ b/include/net/dropreason.h @@ -23,6 +23,12 @@ enum skb_drop_reason_subsys { */ SKB_DROP_REASON_SUBSYS_OPENVSWITCH, + /** + * @SKB_DROP_REASON_SUBSYS_QDISC: TC qdisc drop reasons, + * see include/net/dropreason-qdisc.h + */ + SKB_DROP_REASON_SUBSYS_QDISC, + /** @SKB_DROP_REASON_SUBSYS_NUM: number of subsystems defined */ SKB_DROP_REASON_SUBSYS_NUM }; diff --git a/include/net/sch_generic.h b/include/net/sch_generic.h index c3a7268b567e..f3ee0bd5d0f3 100644 --- a/include/net/sch_generic.h +++ b/include/net/sch_generic.h @@ -20,12 +20,15 @@ #include <net/rtnetlink.h> #include <net/flow_offload.h> #include <linux/xarray.h> +#include <net/dropreason-qdisc.h> struct Qdisc_ops; struct qdisc_walker; struct tcf_walker; struct module; struct bpf_flow_keys; +struct Qdisc; +struct netdev_queue; struct qdisc_rate_table { struct tc_ratespec rate; @@ -1106,36 +1109,43 @@ static inline struct tc_skb_cb *tc_skb_cb(const struct sk_buff *skb) return cb; } +/* TC classifier accessors - use enum skb_drop_reason */ static inline enum skb_drop_reason tcf_get_drop_reason(const struct sk_buff *skb) { - return tc_skb_cb(skb)->drop_reason; + return (enum skb_drop_reason)tc_skb_cb(skb)->drop_reason; } static inline void tcf_set_drop_reason(const struct sk_buff *skb, enum skb_drop_reason reason) { - tc_skb_cb(skb)->drop_reason = reason; + tc_skb_cb(skb)->drop_reason = (enum qdisc_drop_reason)reason; } -static inline void tcf_kfree_skb_list(struct sk_buff *skb) +/* Qdisc accessors - use enum qdisc_drop_reason */ +static inline enum qdisc_drop_reason +tcf_get_qdisc_drop_reason(const struct sk_buff *skb) { - while (unlikely(skb)) { - struct sk_buff *next = skb->next; + return tc_skb_cb(skb)->drop_reason; +} - prefetch(next); - kfree_skb_reason(skb, tcf_get_drop_reason(skb)); - skb = next; - } +static inline void tcf_set_qdisc_drop_reason(const struct sk_buff *skb, + enum qdisc_drop_reason reason) +{ + tc_skb_cb(skb)->drop_reason = reason; } +void tcf_kfree_skb_list(struct sk_buff *skb, struct Qdisc *q, + struct netdev_queue *txq, + struct net_device *dev); + static inline void qdisc_dequeue_drop(struct Qdisc *q, struct sk_buff *skb, - enum skb_drop_reason reason) + enum qdisc_drop_reason reason) { DEBUG_NET_WARN_ON_ONCE(!(q->flags & TCQ_F_DEQUEUE_DROPS)); DEBUG_NET_WARN_ON_ONCE(q->flags & TCQ_F_NOLOCK); - tcf_set_drop_reason(skb, reason); + tcf_set_qdisc_drop_reason(skb, reason); skb->next = q->to_free; q->to_free = skb; } @@ -1312,9 +1322,9 @@ static inline int qdisc_drop(struct sk_buff *skb, struct Qdisc *sch, static inline int qdisc_drop_reason(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free, - enum skb_drop_reason reason) + enum qdisc_drop_reason reason) { - tcf_set_drop_reason(skb, reason); + tcf_set_qdisc_drop_reason(skb, reason); return qdisc_drop(skb, sch, to_free); } diff --git a/include/trace/events/qdisc.h b/include/trace/events/qdisc.h index ff33f41a9db7..d8a5c2677470 100644 --- a/include/trace/events/qdisc.h +++ b/include/trace/events/qdisc.h @@ -74,6 +74,57 @@ TRACE_EVENT(qdisc_enqueue, __entry->ifindex, __entry->handle, __entry->parent, __entry->skbaddr) ); +#undef FN +#undef FNe +#define FN(reason) TRACE_DEFINE_ENUM(QDISC_DROP_##reason); +#define FNe(reason) TRACE_DEFINE_ENUM(QDISC_DROP_##reason); +DEFINE_QDISC_DROP_REASON(FN, FNe) + +#undef FN +#undef FNe +#define FN(reason) { QDISC_DROP_##reason, #reason }, +#define FNe(reason) { QDISC_DROP_##reason, #reason } + +TRACE_EVENT(qdisc_drop, + + TP_PROTO(struct Qdisc *qdisc, const struct netdev_queue *txq, + struct net_device *dev, struct sk_buff *skb, + enum qdisc_drop_reason reason), + + TP_ARGS(qdisc, txq, dev, skb, reason), + + TP_STRUCT__entry( + __field(struct Qdisc *, qdisc) + __field(const struct netdev_queue *, txq) + __field(void *, skbaddr) + __field(int, ifindex) + __field(u32, handle) + __field(u32, parent) + __field(enum qdisc_drop_reason, reason) + __string(kind, qdisc->ops->id) + ), + + TP_fast_assign( + __entry->qdisc = qdisc; + __entry->txq = txq; + __entry->skbaddr = skb; + __entry->ifindex = dev ? dev->ifindex : 0; + __entry->handle = qdisc->handle; + __entry->parent = qdisc->parent; + __entry->reason = reason; + __assign_str(kind); + ), + + TP_printk("drop ifindex=%d kind=%s handle=0x%X parent=0x%X skbaddr=%p reason=%s", + __entry->ifindex, __get_str(kind), __entry->handle, + __entry->parent, __entry->skbaddr, + __print_symbolic(__entry->reason, + DEFINE_QDISC_DROP_REASON(FN, FNe))) +); + +#undef FN +#undef FNe + TRACE_EVENT(qdisc_reset, TP_PROTO(struct Qdisc *q), diff --git a/net/core/dev.c b/net/core/dev.c index 43de5af0d6ec..33262344b751 100644 --- a/net/core/dev.c +++ b/net/core/dev.c @@ -4161,7 +4161,7 @@ static inline int __dev_xmit_skb(struct sk_buff *skb, struct Qdisc *q, qdisc_calculate_pkt_len(skb, q); - tcf_set_drop_reason(skb, SKB_DROP_REASON_QDISC_DROP); + tcf_set_qdisc_drop_reason(skb, QDISC_DROP_GENERIC); if (q->flags & TCQ_F_NOLOCK) { if (q->flags & TCQ_F_CAN_BYPASS && nolock_qdisc_is_empty(q) && @@ -4269,8 +4269,8 @@ static inline int __dev_xmit_skb(struct sk_buff *skb, struct Qdisc *q, spin_unlock(root_lock); free_skbs: - tcf_kfree_skb_list(to_free); - tcf_kfree_skb_list(to_free2); + tcf_kfree_skb_list(to_free, q, txq, dev); + tcf_kfree_skb_list(to_free2, q, txq, dev); return rc; } @@ -5795,7 +5795,7 @@ static __latent_entropy void net_tx_action(void) to_free = qdisc_run(q); if (root_lock) spin_unlock(root_lock); - tcf_kfree_skb_list(to_free); + tcf_kfree_skb_list(to_free, q, NULL, qdisc_dev(q)); } rcu_read_unlock(); diff --git a/net/sched/sch_cake.c b/net/sched/sch_cake.c index fd56b7d88301..a2d1b292600d 100644 --- a/net/sched/sch_cake.c +++ b/net/sched/sch_cake.c @@ -548,7 +548,7 @@ static enum skb_drop_reason cobalt_should_drop(struct cobalt_vars *vars, if (next_due && vars->dropping) { /* Use ECN mark if possible, otherwise drop */ if (!(vars->ecn_marked = INET_ECN_set_ce(skb))) - reason = SKB_DROP_REASON_QDISC_CONGESTED; + reason = QDISC_DROP_CONGESTED; vars->count++; if (!vars->count) @@ -573,7 +573,7 @@ static enum skb_drop_reason cobalt_should_drop(struct cobalt_vars *vars, /* Simple BLUE implementation. Lack of ECN is deliberate. */ if (vars->p_drop && reason == SKB_NOT_DROPPED_YET && get_random_u32() < vars->p_drop) - reason = SKB_DROP_REASON_CAKE_FLOOD; + reason = QDISC_DROP_CAKE_FLOOD; /* Overload the drop_next field as an activity timeout */ if (!vars->count) @@ -1604,7 +1604,7 @@ static unsigned int cake_drop(struct Qdisc *sch, struct sk_buff **to_free) if (q->config->rate_flags & CAKE_FLAG_INGRESS) cake_advance_shaper(q, b, skb, now, true); - qdisc_drop_reason(skb, sch, to_free, SKB_DROP_REASON_QDISC_OVERLIMIT); + qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_OVERLIMIT); sch->q.qlen--; cake_heapify(q, 0); diff --git a/net/sched/sch_codel.c b/net/sched/sch_codel.c index c6551578f1cf..dc2be90666ff 100644 --- a/net/sched/sch_codel.c +++ b/net/sched/sch_codel.c @@ -52,7 +52,7 @@ static void drop_func(struct sk_buff *skb, void *ctx) { struct Qdisc *sch = ctx; - qdisc_dequeue_drop(sch, skb, SKB_DROP_REASON_QDISC_CONGESTED); + qdisc_dequeue_drop(sch, skb, QDISC_DROP_CONGESTED); qdisc_qstats_drop(sch); } @@ -86,8 +86,7 @@ static int codel_qdisc_enqueue(struct sk_buff *skb, struct Qdisc *sch, } q = qdisc_priv(sch); q->drop_overlimit++; - return qdisc_drop_reason(skb, sch, to_free, - SKB_DROP_REASON_QDISC_OVERLIMIT); + return qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_OVERLIMIT); } static const struct nla_policy codel_policy[TCA_CODEL_MAX + 1] = { diff --git a/net/sched/sch_dualpi2.c b/net/sched/sch_dualpi2.c index 6d7e6389758d..0b6a38704a3b 100644 --- a/net/sched/sch_dualpi2.c +++ b/net/sched/sch_dualpi2.c @@ -393,13 +393,11 @@ static int dualpi2_enqueue_skb(struct sk_buff *skb, struct Qdisc *sch, qdisc_qstats_overlimit(sch); if (skb_in_l_queue(skb)) qdisc_qstats_overlimit(q->l_queue); - return qdisc_drop_reason(skb, sch, to_free, - SKB_DROP_REASON_QDISC_OVERLIMIT); + return qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_OVERLIMIT); } if (q->drop_early && must_drop(sch, q, skb)) { - qdisc_drop_reason(skb, sch, to_free, - SKB_DROP_REASON_QDISC_CONGESTED); + qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_CONGESTED); return NET_XMIT_SUCCESS | __NET_XMIT_BYPASS; } @@ -593,7 +591,7 @@ static struct sk_buff *dualpi2_qdisc_dequeue(struct Qdisc *sch) while ((skb = dequeue_packet(sch, q, &credit_change, now))) { if (!q->drop_early && must_drop(sch, q, skb)) { drop_and_retry(q, skb, sch, - SKB_DROP_REASON_QDISC_CONGESTED); + QDISC_DROP_CONGESTED); continue; } diff --git a/net/sched/sch_fq.c b/net/sched/sch_fq.c index d0200ec8ada6..1ae254971326 100644 --- a/net/sched/sch_fq.c +++ b/net/sched/sch_fq.c @@ -541,7 +541,7 @@ static bool fq_packet_beyond_horizon(const struct sk_buff *skb, return unlikely((s64)skb->tstamp > (s64)(now + q->horizon)); } -#define FQDR(reason) SKB_DROP_REASON_FQ_##reason +#define FQDR(reason) QDISC_DROP_FQ_##reason static int fq_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) @@ -554,8 +554,7 @@ static int fq_enqueue(struct sk_buff *skb, struct Qdisc *sch, band = fq_prio2band(q->prio2band, skb->priority & TC_PRIO_MAX); if (unlikely(q->band_pkt_count[band] >= sch->limit)) { q->stat_band_drops[band]++; - return qdisc_drop_reason(skb, sch, to_free, - FQDR(BAND_LIMIT)); + return qdisc_drop_reason(skb, sch, to_free, FQDR(BAND_LIMIT)); } now = ktime_get_ns(); @@ -581,7 +580,7 @@ static int fq_enqueue(struct sk_buff *skb, struct Qdisc *sch, if (unlikely(f->qlen >= q->flow_plimit)) { q->stat_flows_plimit++; return qdisc_drop_reason(skb, sch, to_free, - FQDR(FLOW_LIMIT)); + QDISC_DROP_FQ_FLOW_LIMIT); } if (fq_flow_is_detached(f)) { diff --git a/net/sched/sch_fq_codel.c b/net/sched/sch_fq_codel.c index dc187c7f06b1..3e384a344bc3 100644 --- a/net/sched/sch_fq_codel.c +++ b/net/sched/sch_fq_codel.c @@ -168,7 +168,7 @@ static unsigned int fq_codel_drop(struct Qdisc *sch, unsigned int max_packets, skb = dequeue_head(flow); len += qdisc_pkt_len(skb); mem += get_codel_cb(skb)->mem_usage; - tcf_set_drop_reason(skb, SKB_DROP_REASON_QDISC_OVERLIMIT); + tcf_set_qdisc_drop_reason(skb, QDISC_DROP_OVERLIMIT); __qdisc_drop(skb, to_free); } while (++i < max_packets && len < threshold); @@ -275,7 +275,7 @@ static void drop_func(struct sk_buff *skb, void *ctx) { struct Qdisc *sch = ctx; - qdisc_dequeue_drop(sch, skb, SKB_DROP_REASON_QDISC_CONGESTED); + qdisc_dequeue_drop(sch, skb, QDISC_DROP_CONGESTED); qdisc_qstats_drop(sch); } diff --git a/net/sched/sch_fq_pie.c b/net/sched/sch_fq_pie.c index 7b96bc3ff891..9fe997179d78 100644 --- a/net/sched/sch_fq_pie.c +++ b/net/sched/sch_fq_pie.c @@ -130,7 +130,7 @@ static inline void flow_queue_add(struct fq_pie_flow *flow, static int fq_pie_qdisc_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) { - enum skb_drop_reason reason = SKB_DROP_REASON_QDISC_OVERLIMIT; + enum qdisc_drop_reason reason = QDISC_DROP_OVERLIMIT; struct fq_pie_sched_data *q = qdisc_priv(sch); struct fq_pie_flow *sel_flow; int ret; @@ -162,7 +162,7 @@ static int fq_pie_qdisc_enqueue(struct sk_buff *skb, struct Qdisc *sch, q->overmemory++; } - reason = SKB_DROP_REASON_QDISC_CONGESTED; + reason = QDISC_DROP_CONGESTED; if (!pie_drop_early(sch, &q->p_params, &sel_flow->vars, sel_flow->backlog, skb->len)) { diff --git a/net/sched/sch_generic.c b/net/sched/sch_generic.c index 98ffe64de51f..d938fbd32b26 100644 --- a/net/sched/sch_generic.c +++ b/net/sched/sch_generic.c @@ -25,11 +25,11 @@ #include <linux/skb_array.h> #include <linux/if_macvlan.h> #include <linux/bpf.h> +#include <trace/events/qdisc.h> #include <net/sch_generic.h> #include <net/pkt_sched.h> #include <net/dst.h> #include <net/hotdata.h> -#include <trace/events/qdisc.h> #include <trace/events/net.h> #include <net/xfrm.h> @@ -37,6 +37,31 @@ const struct Qdisc_ops *default_qdisc_ops = &pfifo_fast_ops; EXPORT_SYMBOL(default_qdisc_ops); +void tcf_kfree_skb_list(struct sk_buff *skb, struct Qdisc *q, + struct netdev_queue *txq, struct net_device *dev) +{ + while (unlikely(skb)) { + struct sk_buff *next = skb->next; + u32 reason = tc_skb_cb(skb)->drop_reason; + enum skb_drop_reason skb_reason; + + prefetch(next); + /* TC classifier and qdisc share drop_reason storage. + * Check subsystem mask to identify qdisc drop reasons, + * else pass through skb_drop_reason set by TC classifier. + */ + if ((reason & SKB_DROP_REASON_SUBSYS_MASK) == __QDISC_DROP_REASON) { + trace_qdisc_drop(q, txq, dev, skb, (enum qdisc_drop_reason)reason); + skb_reason = SKB_DROP_REASON_QDISC_DROP; + } else { + skb_reason = (enum skb_drop_reason)reason; + } + kfree_skb_reason(skb, skb_reason); + skb = next; + } +} +EXPORT_SYMBOL(tcf_kfree_skb_list); + static void qdisc_maybe_clear_missed(struct Qdisc *q, const struct netdev_queue *txq) { @@ -741,7 +766,7 @@ static int pfifo_fast_enqueue(struct sk_buff *skb, struct Qdisc *qdisc, err = skb_array_produce(q, skb); if (unlikely(err)) { - tcf_set_drop_reason(skb, SKB_DROP_REASON_QDISC_OVERLIMIT); + tcf_set_qdisc_drop_reason(skb, QDISC_DROP_OVERLIMIT); if (qdisc_is_percpu_stats(qdisc)) return qdisc_drop_cpu(skb, qdisc, to_free); diff --git a/net/sched/sch_gred.c b/net/sched/sch_gred.c index 532fde548b88..66b72a09725f 100644 --- a/net/sched/sch_gred.c +++ b/net/sched/sch_gred.c @@ -251,10 +251,10 @@ static int gred_enqueue(struct sk_buff *skb, struct Qdisc *sch, q->stats.pdrop++; drop: - return qdisc_drop_reason(skb, sch, to_free, SKB_DROP_REASON_QDISC_OVERLIMIT); + return qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_OVERLIMIT); congestion_drop: - qdisc_drop_reason(skb, sch, to_free, SKB_DROP_REASON_QDISC_CONGESTED); + qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_CONGESTED); return NET_XMIT_CN; } diff --git a/net/sched/sch_pie.c b/net/sched/sch_pie.c index 0a377313b6a9..16f3f629cb8e 100644 --- a/net/sched/sch_pie.c +++ b/net/sched/sch_pie.c @@ -85,7 +85,7 @@ EXPORT_SYMBOL_GPL(pie_drop_early); static int pie_qdisc_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) { - enum skb_drop_reason reason = SKB_DROP_REASON_QDISC_OVERLIMIT; + enum qdisc_drop_reason reason = QDISC_DROP_OVERLIMIT; struct pie_sched_data *q = qdisc_priv(sch); bool enqueue = false; @@ -94,7 +94,7 @@ static int pie_qdisc_enqueue(struct sk_buff *skb, struct Qdisc *sch, goto out; } - reason = SKB_DROP_REASON_QDISC_CONGESTED; + reason = QDISC_DROP_CONGESTED; if (!pie_drop_early(sch, &q->params, &q->vars, sch->qstats.backlog, skb->len)) { diff --git a/net/sched/sch_red.c b/net/sched/sch_red.c index 479c42d11083..c8d3d09f15e3 100644 --- a/net/sched/sch_red.c +++ b/net/sched/sch_red.c @@ -70,7 +70,7 @@ static int red_use_nodrop(struct red_sched_data *q) static int red_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) { - enum skb_drop_reason reason = SKB_DROP_REASON_QDISC_CONGESTED; + enum qdisc_drop_reason reason = QDISC_DROP_CONGESTED; struct red_sched_data *q = qdisc_priv(sch); struct Qdisc *child = q->qdisc; unsigned int len; @@ -108,7 +108,7 @@ static int red_enqueue(struct sk_buff *skb, struct Qdisc *sch, break; case RED_HARD_MARK: - reason = SKB_DROP_REASON_QDISC_OVERLIMIT; + reason = QDISC_DROP_OVERLIMIT; qdisc_qstats_overlimit(sch); if (red_use_harddrop(q) || !red_use_ecn(q)) { q->stats.forced_drop++; diff --git a/net/sched/sch_sfb.c b/net/sched/sch_sfb.c index d2835f1168e1..013738662128 100644 --- a/net/sched/sch_sfb.c +++ b/net/sched/sch_sfb.c @@ -280,7 +280,7 @@ static int sfb_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) { - enum skb_drop_reason reason = SKB_DROP_REASON_QDISC_OVERLIMIT; + enum qdisc_drop_reason reason = QDISC_DROP_OVERLIMIT; struct sfb_sched_data *q = qdisc_priv(sch); unsigned int len = qdisc_pkt_len(skb); struct Qdisc *child = q->qdisc; @@ -381,7 +381,7 @@ static int sfb_enqueue(struct sk_buff *skb, struct Qdisc *sch, } r = get_random_u16() & SFB_MAX_PROB; - reason = SKB_DROP_REASON_QDISC_CONGESTED; + reason = QDISC_DROP_CONGESTED; if (unlikely(r < p_min)) { if (unlikely(p_min > SFB_MAX_PROB / 2)) { ^ permalink raw reply related [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing 2026-02-06 16:24 ` [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing Jesper Dangaard Brouer @ 2026-02-10 11:47 ` Toke Høiland-Jørgensen 2026-02-11 8:36 ` Paolo Abeni 2026-02-11 8:50 ` Paolo Abeni 2 siblings, 0 replies; 17+ messages in thread From: Toke Høiland-Jørgensen @ 2026-02-10 11:47 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Paolo Abeni Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Jesper Dangaard Brouer <hawk@kernel.org> writes: > Create new enum qdisc_drop_reason and trace_qdisc_drop tracepoint > for qdisc layer drop diagnostics with direct qdisc context visibility. > > The new tracepoint includes qdisc handle, parent, kind (name), and > device information. Existing SKB_DROP_REASON_QDISC_DROP is retained > for backwards compatibility via kfree_skb_reason(). > > Convert FQ, FQ_CoDel, CoDel, SFB, and pfifo_fast to use the new > infrastructure. > > Reported-by: syzbot+ci335f183b8446659a@syzkaller.appspotmail.com > Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> One small nit, see below. With that: Reviewed-by: Toke Høiland-Jørgensen <toke@redhat.com> [...] > @@ -37,6 +37,31 @@ > const struct Qdisc_ops *default_qdisc_ops = &pfifo_fast_ops; > EXPORT_SYMBOL(default_qdisc_ops); > > +void tcf_kfree_skb_list(struct sk_buff *skb, struct Qdisc *q, > + struct netdev_queue *txq, struct net_device *dev) > +{ > + while (unlikely(skb)) { Seems a bit odd to have the unlikely() here. I guess it was originally there because it was an inlined function, but since it isn't anymore (presumably?), does it really do anything? -Toke ^ permalink raw reply [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing 2026-02-06 16:24 ` [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing Jesper Dangaard Brouer 2026-02-10 11:47 ` Toke Høiland-Jørgensen @ 2026-02-11 8:36 ` Paolo Abeni 2026-02-11 8:46 ` Paolo Abeni 2026-02-11 8:50 ` Paolo Abeni 2 siblings, 1 reply; 17+ messages in thread From: Paolo Abeni @ 2026-02-11 8:36 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Toke Høiland-Jørgensen Cc: bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team On 2/6/26 5:24 PM, Jesper Dangaard Brouer wrote: > diff --git a/net/sched/sch_cake.c b/net/sched/sch_cake.c > index fd56b7d88301..a2d1b292600d 100644 > --- a/net/sched/sch_cake.c > +++ b/net/sched/sch_cake.c > @@ -548,7 +548,7 @@ static enum skb_drop_reason cobalt_should_drop(struct cobalt_vars *vars, > if (next_due && vars->dropping) { > /* Use ECN mark if possible, otherwise drop */ > if (!(vars->ecn_marked = INET_ECN_set_ce(skb))) > - reason = SKB_DROP_REASON_QDISC_CONGESTED; > + reason = QDISC_DROP_CONGESTED; FTR I think an explicit cast here should be better than changing the cobalt_should_drop() return type. > @@ -593,7 +591,7 @@ static struct sk_buff *dualpi2_qdisc_dequeue(struct Qdisc *sch) > while ((skb = dequeue_packet(sch, q, &credit_change, now))) { > if (!q->drop_early && must_drop(sch, q, skb)) { > drop_and_retry(q, skb, sch, > - SKB_DROP_REASON_QDISC_CONGESTED); > + QDISC_DROP_CONGESTED); An explicit cast is needed above to avoid compiler warning (or you can change drop_and_retry() signature). Also a few lines below there is another drop_and_retry() calls that is not converted. > @@ -37,6 +37,31 @@ > const struct Qdisc_ops *default_qdisc_ops = &pfifo_fast_ops; > EXPORT_SYMBOL(default_qdisc_ops); > > +void tcf_kfree_skb_list(struct sk_buff *skb, struct Qdisc *q, > + struct netdev_queue *txq, struct net_device *dev) > +{ > + while (unlikely(skb)) { AFAICS tcf_kfree_skb_list() is invoked in the fastpath even when skb is likely NULL. I guess it would be better to split it an inline helper and an exported symbol: // in sch_generic.h static inline void tcf_kfree_skb_list(struct sk_buff *skb, struct Qdisc *q, struct netdev_queue *txq, struct net_device *dev) { if (unlikely(skb)) __tcf_kfree_skb_list(skb, q, txq, dev); } // in sch_generic.c void __tcf_kfree_skb_list(struct sk_buff *skb, struct Qdisc *q, struct netdev_queue *txq, struct net_device *dev) { while (skb) { //... Hopefully the above should fit the recent compiler oriented optimization done by Eric. /P ^ permalink raw reply [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing 2026-02-11 8:36 ` Paolo Abeni @ 2026-02-11 8:46 ` Paolo Abeni 2026-02-13 17:50 ` Jesper Dangaard Brouer 0 siblings, 1 reply; 17+ messages in thread From: Paolo Abeni @ 2026-02-11 8:46 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Toke Høiland-Jørgensen Cc: bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team On 2/11/26 9:36 AM, Paolo Abeni wrote: > On 2/6/26 5:24 PM, Jesper Dangaard Brouer wrote: >> @@ -593,7 +591,7 @@ static struct sk_buff *dualpi2_qdisc_dequeue(struct Qdisc *sch) >> while ((skb = dequeue_packet(sch, q, &credit_change, now))) { >> if (!q->drop_early && must_drop(sch, q, skb)) { >> drop_and_retry(q, skb, sch, >> - SKB_DROP_REASON_QDISC_CONGESTED); >> + QDISC_DROP_CONGESTED); > > An explicit cast is needed above to avoid compiler warning (or you can > change drop_and_retry() signature). I see that the type confusion issues are handled for good by later patches in the series, but I still think it would be better to avoid introducing them with explicit casts as needed (removed by later patches). > Also a few lines below there is another drop_and_retry() calls that is > not converted. I see it's handled by a later patch. I guess it would be useful to mention such fact in the commit message. Thanks, Paolo ^ permalink raw reply [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing 2026-02-11 8:46 ` Paolo Abeni @ 2026-02-13 17:50 ` Jesper Dangaard Brouer 0 siblings, 0 replies; 17+ messages in thread From: Jesper Dangaard Brouer @ 2026-02-13 17:50 UTC (permalink / raw) To: Paolo Abeni, netdev, Eric Dumazet, David S. Miller, Toke Høiland-Jørgensen Cc: bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team On 11/02/2026 09.46, Paolo Abeni wrote: > On 2/11/26 9:36 AM, Paolo Abeni wrote: >> On 2/6/26 5:24 PM, Jesper Dangaard Brouer wrote: >>> @@ -593,7 +591,7 @@ static struct sk_buff *dualpi2_qdisc_dequeue(struct Qdisc *sch) >>> while ((skb = dequeue_packet(sch, q, &credit_change, now))) { >>> if (!q->drop_early && must_drop(sch, q, skb)) { >>> drop_and_retry(q, skb, sch, >>> - SKB_DROP_REASON_QDISC_CONGESTED); >>> + QDISC_DROP_CONGESTED); >> >> An explicit cast is needed above to avoid compiler warning (or you can >> change drop_and_retry() signature). > > I see that the type confusion issues are handled for good by later > patches in the series, but I still think it would be better to avoid > introducing them with explicit casts as needed (removed by later patches). > >> Also a few lines below there is another drop_and_retry() calls that is >> not converted. > > I see it's handled by a later patch. I guess it would be useful to > mention such fact in the commit message. Yes, this is handled in patch 3/6. I acknowledge this is annoying from a review perspective, and also wrong because I'm leaving compiler warning that I'm fixing up later. The reason for splitting it out into a separate patch was because I wanted document (in commit desc) the special usage of SKB_NOT_DROPPED_YET as a sentinel value. I'm leaning towards folding patch 3 into this patch 1, as suggested by Toke. --Jesper ^ permalink raw reply [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing 2026-02-06 16:24 ` [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing Jesper Dangaard Brouer 2026-02-10 11:47 ` Toke Høiland-Jørgensen 2026-02-11 8:36 ` Paolo Abeni @ 2026-02-11 8:50 ` Paolo Abeni 2 siblings, 0 replies; 17+ messages in thread From: Paolo Abeni @ 2026-02-11 8:50 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Toke Høiland-Jørgensen Cc: bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team On 2/6/26 5:24 PM, Jesper Dangaard Brouer wrote: > Create new enum qdisc_drop_reason and trace_qdisc_drop tracepoint > for qdisc layer drop diagnostics with direct qdisc context visibility. > > The new tracepoint includes qdisc handle, parent, kind (name), and > device information. Existing SKB_DROP_REASON_QDISC_DROP is retained > for backwards compatibility via kfree_skb_reason(). > > Convert FQ, FQ_CoDel, CoDel, SFB, and pfifo_fast to use the new > infrastructure. > > Reported-by: syzbot+ci335f183b8446659a@syzkaller.appspotmail.com > Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> > --- > include/net/dropreason-core.h | 42 ++---------------- > include/net/dropreason-qdisc.h | 95 ++++++++++++++++++++++++++++++++++++++++ I think it would be useful to add dropreason-qdisc.h to the net/sched maintainer's entry. /P ^ permalink raw reply [flat|nested] 17+ messages in thread
* [PATCH net-next v3 2/6] net: sched: sfq: convert to qdisc drop reasons 2026-02-06 16:24 [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint Jesper Dangaard Brouer 2026-02-06 16:24 ` [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing Jesper Dangaard Brouer @ 2026-02-06 16:24 ` Jesper Dangaard Brouer 2026-02-10 11:48 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 3/6] net: sched: sch_cake: use enum qdisc_drop_reason for cobalt_should_drop Jesper Dangaard Brouer ` (3 subsequent siblings) 5 siblings, 1 reply; 17+ messages in thread From: Jesper Dangaard Brouer @ 2026-02-06 16:24 UTC (permalink / raw) To: netdev, Eric Dumazet, David S. Miller, Paolo Abeni, Toke Høiland-Jørgensen Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Convert SFQ to use the new qdisc-specific drop reason infrastructure. This patch demonstrates how to convert a flow-based qdisc to use the new enum qdisc_drop_reason. As part of this conversion: - Add QDISC_DROP_MAXFLOWS for flow table exhaustion - Rename FQ_FLOW_LIMIT to generic FLOW_LIMIT, now shared by FQ and SFQ - Use QDISC_DROP_OVERLIMIT for sfq_drop() when overall limit exceeded - Use QDISC_DROP_FLOW_LIMIT for per-flow depth limit exceeded The FLOW_LIMIT reason is now a common drop reason for per-flow limits, applicable to both FQ and SFQ qdiscs. Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> --- include/net/dropreason-qdisc.h | 22 ++++++++++++++++------ net/sched/sch_fq.c | 2 +- net/sched/sch_sfq.c | 8 ++++---- 3 files changed, 21 insertions(+), 11 deletions(-) diff --git a/include/net/dropreason-qdisc.h b/include/net/dropreason-qdisc.h index db81b3471a65..b019a3a0a59b 100644 --- a/include/net/dropreason-qdisc.h +++ b/include/net/dropreason-qdisc.h @@ -9,10 +9,11 @@ FN(GENERIC) \ FN(OVERLIMIT) \ FN(CONGESTED) \ + FN(FLOW_LIMIT) \ + FN(MAXFLOWS) \ FN(CAKE_FLOOD) \ FN(FQ_BAND_LIMIT) \ FN(FQ_HORIZON_LIMIT) \ - FN(FQ_FLOW_LIMIT) \ FNe(MAX) #undef FN @@ -59,6 +60,20 @@ enum qdisc_drop_reason { * congestion to the sender and prevent bufferbloat. */ QDISC_DROP_CONGESTED, + /** + * @QDISC_DROP_FLOW_LIMIT: packet dropped because an individual flow + * exceeded its per-flow packet/depth limit. Used by FQ and SFQ qdiscs + * to enforce per-flow fairness and prevent a single flow from + * monopolizing queue resources. + */ + QDISC_DROP_FLOW_LIMIT, + /** + * @QDISC_DROP_MAXFLOWS: packet dropped because the qdisc's flow + * tracking table is full and no free slots are available to allocate + * for a new flow. This indicates flow table exhaustion in flow-based + * qdiscs that maintain per-flow state (e.g., SFQ). + */ + QDISC_DROP_MAXFLOWS, /** * @QDISC_DROP_CAKE_FLOOD: CAKE qdisc dropped packet due to flood * protection mechanism (BLUE algorithm). This indicates potential @@ -77,11 +92,6 @@ enum qdisc_drop_reason { * packets with unreasonable future timestamps from blocking the queue. */ QDISC_DROP_FQ_HORIZON_LIMIT, - /** - * @QDISC_DROP_FQ_FLOW_LIMIT: FQ dropped packet because an individual - * flow exceeded its per-flow packet limit. - */ - QDISC_DROP_FQ_FLOW_LIMIT, /** * @QDISC_DROP_MAX: the maximum of qdisc drop reasons, which * shouldn't be used as a real 'reason' - only for tracing code gen diff --git a/net/sched/sch_fq.c b/net/sched/sch_fq.c index 1ae254971326..d36da3918b55 100644 --- a/net/sched/sch_fq.c +++ b/net/sched/sch_fq.c @@ -580,7 +580,7 @@ static int fq_enqueue(struct sk_buff *skb, struct Qdisc *sch, if (unlikely(f->qlen >= q->flow_plimit)) { q->stat_flows_plimit++; return qdisc_drop_reason(skb, sch, to_free, - QDISC_DROP_FQ_FLOW_LIMIT); + QDISC_DROP_FLOW_LIMIT); } if (fq_flow_is_detached(f)) { diff --git a/net/sched/sch_sfq.c b/net/sched/sch_sfq.c index 96eb2f122973..efb796976a5b 100644 --- a/net/sched/sch_sfq.c +++ b/net/sched/sch_sfq.c @@ -302,7 +302,7 @@ static unsigned int sfq_drop(struct Qdisc *sch, struct sk_buff **to_free) sfq_dec(q, x); sch->q.qlen--; qdisc_qstats_backlog_dec(sch, skb); - qdisc_drop(skb, sch, to_free); + qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_OVERLIMIT); return len; } @@ -363,7 +363,7 @@ sfq_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) if (x == SFQ_EMPTY_SLOT) { x = q->dep[0].next; /* get a free slot */ if (x >= SFQ_MAX_FLOWS) - return qdisc_drop(skb, sch, to_free); + return qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_MAXFLOWS); q->ht[hash] = x; slot = &q->slots[x]; slot->hash = hash; @@ -420,14 +420,14 @@ sfq_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) if (slot->qlen >= q->maxdepth) { congestion_drop: if (!sfq_headdrop(q)) - return qdisc_drop(skb, sch, to_free); + return qdisc_drop_reason(skb, sch, to_free, QDISC_DROP_FLOW_LIMIT); /* We know we have at least one packet in queue */ head = slot_dequeue_head(slot); delta = qdisc_pkt_len(head) - qdisc_pkt_len(skb); sch->qstats.backlog -= delta; slot->backlog -= delta; - qdisc_drop(head, sch, to_free); + qdisc_drop_reason(head, sch, to_free, QDISC_DROP_FLOW_LIMIT); slot_queue_add(slot, skb); qdisc_tree_reduce_backlog(sch, 0, delta); ^ permalink raw reply related [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 2/6] net: sched: sfq: convert to qdisc drop reasons 2026-02-06 16:24 ` [PATCH net-next v3 2/6] net: sched: sfq: convert to qdisc drop reasons Jesper Dangaard Brouer @ 2026-02-10 11:48 ` Toke Høiland-Jørgensen 0 siblings, 0 replies; 17+ messages in thread From: Toke Høiland-Jørgensen @ 2026-02-10 11:48 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Paolo Abeni Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Jesper Dangaard Brouer <hawk@kernel.org> writes: > Convert SFQ to use the new qdisc-specific drop reason infrastructure. > > This patch demonstrates how to convert a flow-based qdisc to use the > new enum qdisc_drop_reason. As part of this conversion: > > - Add QDISC_DROP_MAXFLOWS for flow table exhaustion > - Rename FQ_FLOW_LIMIT to generic FLOW_LIMIT, now shared by FQ and SFQ > - Use QDISC_DROP_OVERLIMIT for sfq_drop() when overall limit exceeded > - Use QDISC_DROP_FLOW_LIMIT for per-flow depth limit exceeded > > The FLOW_LIMIT reason is now a common drop reason for per-flow limits, > applicable to both FQ and SFQ qdiscs. > > Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> Reviewed-by: Toke Høiland-Jørgensen <toke@redhat.com> ^ permalink raw reply [flat|nested] 17+ messages in thread
* [PATCH net-next v3 3/6] net: sched: sch_cake: use enum qdisc_drop_reason for cobalt_should_drop 2026-02-06 16:24 [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint Jesper Dangaard Brouer 2026-02-06 16:24 ` [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing Jesper Dangaard Brouer 2026-02-06 16:24 ` [PATCH net-next v3 2/6] net: sched: sfq: convert to qdisc drop reasons Jesper Dangaard Brouer @ 2026-02-06 16:24 ` Jesper Dangaard Brouer 2026-02-10 11:52 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 4/6] net: sched: rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION Jesper Dangaard Brouer ` (2 subsequent siblings) 5 siblings, 1 reply; 17+ messages in thread From: Jesper Dangaard Brouer @ 2026-02-06 16:24 UTC (permalink / raw) To: netdev, Eric Dumazet, David S. Miller, Paolo Abeni, Toke Høiland-Jørgensen Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Change cobalt_should_drop() return type from enum skb_drop_reason to enum qdisc_drop_reason to fix implicit enum conversion warnings. Use QDISC_DROP_UNSPEC as the 'not dropped' sentinel instead of SKB_NOT_DROPPED_YET. Both have the same compiled value (0), so the comparison logic remains semantically equivalent. Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> --- net/sched/sch_cake.c | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/net/sched/sch_cake.c b/net/sched/sch_cake.c index a2d1b292600d..da3c347574d2 100644 --- a/net/sched/sch_cake.c +++ b/net/sched/sch_cake.c @@ -497,13 +497,13 @@ static bool cobalt_queue_empty(struct cobalt_vars *vars, /* Call this with a freshly dequeued packet for possible congestion marking. * Returns true as an instruction to drop the packet, false for delivery. */ -static enum skb_drop_reason cobalt_should_drop(struct cobalt_vars *vars, - struct cobalt_params *p, - ktime_t now, - struct sk_buff *skb, - u32 bulk_flows) +static enum qdisc_drop_reason cobalt_should_drop(struct cobalt_vars *vars, + struct cobalt_params *p, + ktime_t now, + struct sk_buff *skb, + u32 bulk_flows) { - enum skb_drop_reason reason = SKB_NOT_DROPPED_YET; + enum qdisc_drop_reason reason = QDISC_DROP_UNSPEC; bool next_due, over_target; ktime_t schedule; u64 sojourn; @@ -571,14 +571,14 @@ static enum skb_drop_reason cobalt_should_drop(struct cobalt_vars *vars, } /* Simple BLUE implementation. Lack of ECN is deliberate. */ - if (vars->p_drop && reason == SKB_NOT_DROPPED_YET && + if (vars->p_drop && reason == QDISC_DROP_UNSPEC && get_random_u32() < vars->p_drop) reason = QDISC_DROP_CAKE_FLOOD; /* Overload the drop_next field as an activity timeout */ if (!vars->count) vars->drop_next = ktime_add_ns(now, p->interval); - else if (ktime_to_ns(schedule) > 0 && reason == SKB_NOT_DROPPED_YET) + else if (ktime_to_ns(schedule) > 0 && reason == QDISC_DROP_UNSPEC) vars->drop_next = now; return reason; @@ -2004,7 +2004,7 @@ static struct sk_buff *cake_dequeue(struct Qdisc *sch) { struct cake_sched_data *q = qdisc_priv(sch); struct cake_tin_data *b = &q->tins[q->cur_tin]; - enum skb_drop_reason reason; + enum qdisc_drop_reason reason; ktime_t now = ktime_get(); struct cake_flow *flow; struct list_head *head; @@ -2225,7 +2225,7 @@ static struct sk_buff *cake_dequeue(struct Qdisc *sch) !!(q->config->rate_flags & CAKE_FLAG_INGRESS))); /* Last packet in queue may be marked, shouldn't be dropped */ - if (reason == SKB_NOT_DROPPED_YET || !flow->head) + if (reason == QDISC_DROP_UNSPEC || !flow->head) break; /* drop this packet, get another one */ ^ permalink raw reply related [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 3/6] net: sched: sch_cake: use enum qdisc_drop_reason for cobalt_should_drop 2026-02-06 16:24 ` [PATCH net-next v3 3/6] net: sched: sch_cake: use enum qdisc_drop_reason for cobalt_should_drop Jesper Dangaard Brouer @ 2026-02-10 11:52 ` Toke Høiland-Jørgensen 0 siblings, 0 replies; 17+ messages in thread From: Toke Høiland-Jørgensen @ 2026-02-10 11:52 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Paolo Abeni Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Jesper Dangaard Brouer <hawk@kernel.org> writes: > Change cobalt_should_drop() return type from enum skb_drop_reason to > enum qdisc_drop_reason to fix implicit enum conversion warnings. > > Use QDISC_DROP_UNSPEC as the 'not dropped' sentinel instead of > SKB_NOT_DROPPED_YET. Both have the same compiled value (0), so the > comparison logic remains semantically equivalent. > > Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> So before this patch (but after patch 1), we're basically assigning invalid values to the enum skb_drop_reason variable? I guess the compiler just turns everything into their integer representations, since this doesn't cause build errors? I guess the only way to avoid this is to fold this patch into patch 1? Not a huge deal, but maybe that's better then? In any case, for the change itself: Reviewed-by: Toke Høiland-Jørgensen <toke@redhat.com> ^ permalink raw reply [flat|nested] 17+ messages in thread
* [PATCH net-next v3 4/6] net: sched: rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION 2026-02-06 16:24 [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint Jesper Dangaard Brouer ` (2 preceding siblings ...) 2026-02-06 16:24 ` [PATCH net-next v3 3/6] net: sched: sch_cake: use enum qdisc_drop_reason for cobalt_should_drop Jesper Dangaard Brouer @ 2026-02-06 16:24 ` Jesper Dangaard Brouer 2026-02-10 11:53 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 5/6] net: sched: rename QDISC_DROP_FQ_* to generic names Jesper Dangaard Brouer 2026-02-06 16:24 ` [PATCH net-next v3 6/6] net: sched: sch_dualpi2: use qdisc_dequeue_drop() for dequeue drops Jesper Dangaard Brouer 5 siblings, 1 reply; 17+ messages in thread From: Jesper Dangaard Brouer @ 2026-02-06 16:24 UTC (permalink / raw) To: netdev, Eric Dumazet, David S. Miller, Paolo Abeni, Toke Høiland-Jørgensen Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION to use a generic name without embedding the qdisc name. This follows the principle that drop reasons should describe the drop mechanism rather than being tied to a specific qdisc implementation. The flood protection drop reason is used by qdiscs implementing probabilistic drop algorithms (like BLUE) that detect unresponsive flows indicating potential DoS or flood attacks. CAKE uses this via its Cobalt AQM component. Suggested-by: Toke Høiland-Jørgensen <toke@redhat.com> Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> --- include/net/dropreason-qdisc.h | 11 ++++++----- net/sched/sch_cake.c | 2 +- 2 files changed, 7 insertions(+), 6 deletions(-) diff --git a/include/net/dropreason-qdisc.h b/include/net/dropreason-qdisc.h index b019a3a0a59b..2e2f4fc36e0d 100644 --- a/include/net/dropreason-qdisc.h +++ b/include/net/dropreason-qdisc.h @@ -11,7 +11,7 @@ FN(CONGESTED) \ FN(FLOW_LIMIT) \ FN(MAXFLOWS) \ - FN(CAKE_FLOOD) \ + FN(FLOOD_PROTECTION) \ FN(FQ_BAND_LIMIT) \ FN(FQ_HORIZON_LIMIT) \ FNe(MAX) @@ -75,11 +75,12 @@ enum qdisc_drop_reason { */ QDISC_DROP_MAXFLOWS, /** - * @QDISC_DROP_CAKE_FLOOD: CAKE qdisc dropped packet due to flood - * protection mechanism (BLUE algorithm). This indicates potential - * DoS/flood attack or unresponsive flow behavior. + * @QDISC_DROP_FLOOD_PROTECTION: packet dropped by flood protection + * mechanism detecting unresponsive flows (potential DoS/flood). + * Used by qdiscs implementing probabilistic drop algorithms like + * BLUE (e.g., CAKE's Cobalt AQM). */ - QDISC_DROP_CAKE_FLOOD, + QDISC_DROP_FLOOD_PROTECTION, /** * @QDISC_DROP_FQ_BAND_LIMIT: FQ (Fair Queue) dropped packet because * the priority band's packet limit was reached. Each priority band diff --git a/net/sched/sch_cake.c b/net/sched/sch_cake.c index da3c347574d2..b08711fa5f97 100644 --- a/net/sched/sch_cake.c +++ b/net/sched/sch_cake.c @@ -573,7 +573,7 @@ static enum qdisc_drop_reason cobalt_should_drop(struct cobalt_vars *vars, /* Simple BLUE implementation. Lack of ECN is deliberate. */ if (vars->p_drop && reason == QDISC_DROP_UNSPEC && get_random_u32() < vars->p_drop) - reason = QDISC_DROP_CAKE_FLOOD; + reason = QDISC_DROP_FLOOD_PROTECTION; /* Overload the drop_next field as an activity timeout */ if (!vars->count) ^ permalink raw reply related [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 4/6] net: sched: rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION 2026-02-06 16:24 ` [PATCH net-next v3 4/6] net: sched: rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION Jesper Dangaard Brouer @ 2026-02-10 11:53 ` Toke Høiland-Jørgensen 0 siblings, 0 replies; 17+ messages in thread From: Toke Høiland-Jørgensen @ 2026-02-10 11:53 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Paolo Abeni Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Jesper Dangaard Brouer <hawk@kernel.org> writes: > Rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION to use a > generic name without embedding the qdisc name. This follows the > principle that drop reasons should describe the drop mechanism rather > than being tied to a specific qdisc implementation. > > The flood protection drop reason is used by qdiscs implementing > probabilistic drop algorithms (like BLUE) that detect unresponsive > flows indicating potential DoS or flood attacks. CAKE uses this via > its Cobalt AQM component. > > Suggested-by: Toke Høiland-Jørgensen <toke@redhat.com> > Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> Reviewed-by: Toke Høiland-Jørgensen <toke@redhat.com> ^ permalink raw reply [flat|nested] 17+ messages in thread
* [PATCH net-next v3 5/6] net: sched: rename QDISC_DROP_FQ_* to generic names 2026-02-06 16:24 [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint Jesper Dangaard Brouer ` (3 preceding siblings ...) 2026-02-06 16:24 ` [PATCH net-next v3 4/6] net: sched: rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION Jesper Dangaard Brouer @ 2026-02-06 16:24 ` Jesper Dangaard Brouer 2026-02-10 11:54 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 6/6] net: sched: sch_dualpi2: use qdisc_dequeue_drop() for dequeue drops Jesper Dangaard Brouer 5 siblings, 1 reply; 17+ messages in thread From: Jesper Dangaard Brouer @ 2026-02-06 16:24 UTC (permalink / raw) To: netdev, Eric Dumazet, David S. Miller, Paolo Abeni, Toke Høiland-Jørgensen Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Rename FQ-specific drop reasons to generic names: - QDISC_DROP_FQ_BAND_LIMIT -> QDISC_DROP_BAND_LIMIT - QDISC_DROP_FQ_HORIZON_LIMIT -> QDISC_DROP_HORIZON_LIMIT This follows the principle that drop reasons should describe the drop mechanism rather than being tied to a specific qdisc implementation. These concepts (priority band limits, timestamp horizon) could apply to other qdiscs as well. Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> --- include/net/dropreason-qdisc.h | 21 +++++++++++---------- net/sched/sch_fq.c | 2 +- 2 files changed, 12 insertions(+), 11 deletions(-) diff --git a/include/net/dropreason-qdisc.h b/include/net/dropreason-qdisc.h index 2e2f4fc36e0d..8661797193b1 100644 --- a/include/net/dropreason-qdisc.h +++ b/include/net/dropreason-qdisc.h @@ -12,8 +12,8 @@ FN(FLOW_LIMIT) \ FN(MAXFLOWS) \ FN(FLOOD_PROTECTION) \ - FN(FQ_BAND_LIMIT) \ - FN(FQ_HORIZON_LIMIT) \ + FN(BAND_LIMIT) \ + FN(HORIZON_LIMIT) \ FNe(MAX) #undef FN @@ -82,17 +82,18 @@ enum qdisc_drop_reason { */ QDISC_DROP_FLOOD_PROTECTION, /** - * @QDISC_DROP_FQ_BAND_LIMIT: FQ (Fair Queue) dropped packet because - * the priority band's packet limit was reached. Each priority band - * in FQ has its own limit. + * @QDISC_DROP_BAND_LIMIT: packet dropped because the priority band's + * limit was reached. Used by qdiscs with priority bands that have + * per-band packet limits (e.g., FQ). */ - QDISC_DROP_FQ_BAND_LIMIT, + QDISC_DROP_BAND_LIMIT, /** - * @QDISC_DROP_FQ_HORIZON_LIMIT: FQ dropped packet because its - * timestamp is too far in the future (beyond horizon). This prevents - * packets with unreasonable future timestamps from blocking the queue. + * @QDISC_DROP_HORIZON_LIMIT: packet dropped because its timestamp + * is too far in the future (beyond horizon). This prevents packets + * with unreasonable future timestamps from occupying queue space. + * Used by qdiscs with time-based scheduling (e.g., FQ). */ - QDISC_DROP_FQ_HORIZON_LIMIT, + QDISC_DROP_HORIZON_LIMIT, /** * @QDISC_DROP_MAX: the maximum of qdisc drop reasons, which * shouldn't be used as a real 'reason' - only for tracing code gen diff --git a/net/sched/sch_fq.c b/net/sched/sch_fq.c index d36da3918b55..f8f360d28da0 100644 --- a/net/sched/sch_fq.c +++ b/net/sched/sch_fq.c @@ -541,7 +541,7 @@ static bool fq_packet_beyond_horizon(const struct sk_buff *skb, return unlikely((s64)skb->tstamp > (s64)(now + q->horizon)); } -#define FQDR(reason) QDISC_DROP_FQ_##reason +#define FQDR(reason) QDISC_DROP_##reason static int fq_enqueue(struct sk_buff *skb, struct Qdisc *sch, struct sk_buff **to_free) ^ permalink raw reply related [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 5/6] net: sched: rename QDISC_DROP_FQ_* to generic names 2026-02-06 16:24 ` [PATCH net-next v3 5/6] net: sched: rename QDISC_DROP_FQ_* to generic names Jesper Dangaard Brouer @ 2026-02-10 11:54 ` Toke Høiland-Jørgensen 0 siblings, 0 replies; 17+ messages in thread From: Toke Høiland-Jørgensen @ 2026-02-10 11:54 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Paolo Abeni Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Jesper Dangaard Brouer <hawk@kernel.org> writes: > Rename FQ-specific drop reasons to generic names: > - QDISC_DROP_FQ_BAND_LIMIT -> QDISC_DROP_BAND_LIMIT > - QDISC_DROP_FQ_HORIZON_LIMIT -> QDISC_DROP_HORIZON_LIMIT > > This follows the principle that drop reasons should describe the drop > mechanism rather than being tied to a specific qdisc implementation. > These concepts (priority band limits, timestamp horizon) could apply > to other qdiscs as well. > > Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> > --- > include/net/dropreason-qdisc.h | 21 +++++++++++---------- > net/sched/sch_fq.c | 2 +- > 2 files changed, 12 insertions(+), 11 deletions(-) > > diff --git a/include/net/dropreason-qdisc.h b/include/net/dropreason-qdisc.h > index 2e2f4fc36e0d..8661797193b1 100644 > --- a/include/net/dropreason-qdisc.h > +++ b/include/net/dropreason-qdisc.h > @@ -12,8 +12,8 @@ > FN(FLOW_LIMIT) \ > FN(MAXFLOWS) \ > FN(FLOOD_PROTECTION) \ > - FN(FQ_BAND_LIMIT) \ > - FN(FQ_HORIZON_LIMIT) \ > + FN(BAND_LIMIT) \ > + FN(HORIZON_LIMIT) \ > FNe(MAX) > > #undef FN > @@ -82,17 +82,18 @@ enum qdisc_drop_reason { > */ > QDISC_DROP_FLOOD_PROTECTION, > /** > - * @QDISC_DROP_FQ_BAND_LIMIT: FQ (Fair Queue) dropped packet because > - * the priority band's packet limit was reached. Each priority band > - * in FQ has its own limit. > + * @QDISC_DROP_BAND_LIMIT: packet dropped because the priority band's > + * limit was reached. Used by qdiscs with priority bands that have > + * per-band packet limits (e.g., FQ). > */ > - QDISC_DROP_FQ_BAND_LIMIT, > + QDISC_DROP_BAND_LIMIT, > /** > - * @QDISC_DROP_FQ_HORIZON_LIMIT: FQ dropped packet because its > - * timestamp is too far in the future (beyond horizon). This prevents > - * packets with unreasonable future timestamps from blocking the queue. > + * @QDISC_DROP_HORIZON_LIMIT: packet dropped because its timestamp > + * is too far in the future (beyond horizon). This prevents packets > + * with unreasonable future timestamps from occupying queue space. Why change "blocking the queue" to "occupying queue space"? The former seems worse, so worth calling out? -Toke ^ permalink raw reply [flat|nested] 17+ messages in thread
* [PATCH net-next v3 6/6] net: sched: sch_dualpi2: use qdisc_dequeue_drop() for dequeue drops 2026-02-06 16:24 [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint Jesper Dangaard Brouer ` (4 preceding siblings ...) 2026-02-06 16:24 ` [PATCH net-next v3 5/6] net: sched: rename QDISC_DROP_FQ_* to generic names Jesper Dangaard Brouer @ 2026-02-06 16:24 ` Jesper Dangaard Brouer 2026-02-10 12:00 ` Toke Høiland-Jørgensen 5 siblings, 1 reply; 17+ messages in thread From: Jesper Dangaard Brouer @ 2026-02-06 16:24 UTC (permalink / raw) To: netdev, Eric Dumazet, David S. Miller, Paolo Abeni, Toke Høiland-Jørgensen Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team DualPI2 drops packets during dequeue but was using kfree_skb_reason() directly, bypassing trace_qdisc_drop. Convert to qdisc_dequeue_drop() and add QDISC_DROP_L4S_STEP_NON_ECN to the qdisc drop reason enum. - Set TCQ_F_DEQUEUE_DROPS flag in dualpi2_init() - Use enum qdisc_drop_reason in drop_and_retry() - Replace kfree_skb_reason() with qdisc_dequeue_drop() Reported-by: kernel test robot <lkp@intel.com> Closes: https://lore.kernel.org/oe-kbuild-all/202602061240.CRDtcOeX-lkp@intel.com/ Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> --- include/net/dropreason-core.h | 6 ------ include/net/dropreason-qdisc.h | 8 ++++++++ net/sched/sch_dualpi2.c | 12 ++++++------ 3 files changed, 14 insertions(+), 12 deletions(-) diff --git a/include/net/dropreason-core.h b/include/net/dropreason-core.h index 3d8d284e05c8..5c8c2eb3d2c5 100644 --- a/include/net/dropreason-core.h +++ b/include/net/dropreason-core.h @@ -121,7 +121,6 @@ FN(CANFD_RX_INVALID_FRAME) \ FN(CANXL_RX_INVALID_FRAME) \ FN(PFMEMALLOC) \ - FN(DUALPI2_STEP_DROP) \ FN(PSP_INPUT) \ FN(PSP_OUTPUT) \ FNe(MAX) @@ -579,11 +578,6 @@ enum skb_drop_reason { * reached a path or socket not eligible for use of memory reserves */ SKB_DROP_REASON_PFMEMALLOC, - /** - * @SKB_DROP_REASON_DUALPI2_STEP_DROP: dropped by the step drop - * threshold of DualPI2 qdisc. - */ - SKB_DROP_REASON_DUALPI2_STEP_DROP, /** @SKB_DROP_REASON_PSP_INPUT: PSP input checks failed */ SKB_DROP_REASON_PSP_INPUT, /** @SKB_DROP_REASON_PSP_OUTPUT: PSP output checks failed */ diff --git a/include/net/dropreason-qdisc.h b/include/net/dropreason-qdisc.h index 8661797193b1..c08ecce2d76b 100644 --- a/include/net/dropreason-qdisc.h +++ b/include/net/dropreason-qdisc.h @@ -14,6 +14,7 @@ FN(FLOOD_PROTECTION) \ FN(BAND_LIMIT) \ FN(HORIZON_LIMIT) \ + FN(L4S_STEP_NON_ECN) \ FNe(MAX) #undef FN @@ -94,6 +95,13 @@ enum qdisc_drop_reason { * Used by qdiscs with time-based scheduling (e.g., FQ). */ QDISC_DROP_HORIZON_LIMIT, + /** + * @QDISC_DROP_L4S_STEP_NON_ECN: DualPI2 qdisc dropped a non-ECN-capable + * packet because the L4S queue delay exceeded the step threshold. + * Since the packet cannot be ECN-marked, it must be dropped to signal + * congestion. See RFC 9332 for the DualQ Coupled AQM step mechanism. + */ + QDISC_DROP_L4S_STEP_NON_ECN, /** * @QDISC_DROP_MAX: the maximum of qdisc drop reasons, which * shouldn't be used as a real 'reason' - only for tracing code gen diff --git a/net/sched/sch_dualpi2.c b/net/sched/sch_dualpi2.c index 0b6a38704a3b..fe6f5e889625 100644 --- a/net/sched/sch_dualpi2.c +++ b/net/sched/sch_dualpi2.c @@ -571,11 +571,11 @@ static int do_step_aqm(struct dualpi2_sched_data *q, struct sk_buff *skb, } static void drop_and_retry(struct dualpi2_sched_data *q, struct sk_buff *skb, - struct Qdisc *sch, enum skb_drop_reason reason) + struct Qdisc *sch, enum qdisc_drop_reason reason) { ++q->deferred_drops_cnt; q->deferred_drops_len += qdisc_pkt_len(skb); - kfree_skb_reason(skb, reason); + qdisc_dequeue_drop(sch, skb, reason); qdisc_qstats_drop(sch); } @@ -590,15 +590,13 @@ static struct sk_buff *dualpi2_qdisc_dequeue(struct Qdisc *sch) while ((skb = dequeue_packet(sch, q, &credit_change, now))) { if (!q->drop_early && must_drop(sch, q, skb)) { - drop_and_retry(q, skb, sch, - QDISC_DROP_CONGESTED); + drop_and_retry(q, skb, sch, QDISC_DROP_CONGESTED); continue; } if (skb_in_l_queue(skb) && do_step_aqm(q, skb, now)) { qdisc_qstats_drop(q->l_queue); - drop_and_retry(q, skb, sch, - SKB_DROP_REASON_DUALPI2_STEP_DROP); + drop_and_retry(q, skb, sch, QDISC_DROP_L4S_STEP_NON_ECN); continue; } @@ -915,6 +913,8 @@ static int dualpi2_init(struct Qdisc *sch, struct nlattr *opt, struct dualpi2_sched_data *q = qdisc_priv(sch); int err; + sch->flags |= TCQ_F_DEQUEUE_DROPS; + q->l_queue = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops, TC_H_MAKE(sch->handle, 1), extack); if (!q->l_queue) ^ permalink raw reply related [flat|nested] 17+ messages in thread
* Re: [PATCH net-next v3 6/6] net: sched: sch_dualpi2: use qdisc_dequeue_drop() for dequeue drops 2026-02-06 16:24 ` [PATCH net-next v3 6/6] net: sched: sch_dualpi2: use qdisc_dequeue_drop() for dequeue drops Jesper Dangaard Brouer @ 2026-02-10 12:00 ` Toke Høiland-Jørgensen 0 siblings, 0 replies; 17+ messages in thread From: Toke Høiland-Jørgensen @ 2026-02-10 12:00 UTC (permalink / raw) To: Jesper Dangaard Brouer, netdev, Eric Dumazet, David S. Miller, Paolo Abeni Cc: Jesper Dangaard Brouer, bpf, Jakub Kicinski, horms, jiri, edumazet, xiyou.wangcong, jhs, atenart, carges, kernel-team Jesper Dangaard Brouer <hawk@kernel.org> writes: > DualPI2 drops packets during dequeue but was using kfree_skb_reason() > directly, bypassing trace_qdisc_drop. Convert to qdisc_dequeue_drop() > and add QDISC_DROP_L4S_STEP_NON_ECN to the qdisc drop reason enum. > > - Set TCQ_F_DEQUEUE_DROPS flag in dualpi2_init() > - Use enum qdisc_drop_reason in drop_and_retry() > - Replace kfree_skb_reason() with qdisc_dequeue_drop() > > Reported-by: kernel test robot <lkp@intel.com> > Closes: https://lore.kernel.org/oe-kbuild-all/202602061240.CRDtcOeX-lkp@intel.com/ > Signed-off-by: Jesper Dangaard Brouer <hawk@kernel.org> Reviewed-by: Toke Høiland-Jørgensen <toke@redhat.com> ^ permalink raw reply [flat|nested] 17+ messages in thread
end of thread, other threads:[~2026-02-13 17:50 UTC | newest] Thread overview: 17+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2026-02-06 16:24 [PATCH net-next v3 0/6] net: sched: refactor qdisc drop reasons into dedicated tracepoint Jesper Dangaard Brouer 2026-02-06 16:24 ` [PATCH net-next v3 1/6] net: sched: introduce qdisc-specific drop reason tracing Jesper Dangaard Brouer 2026-02-10 11:47 ` Toke Høiland-Jørgensen 2026-02-11 8:36 ` Paolo Abeni 2026-02-11 8:46 ` Paolo Abeni 2026-02-13 17:50 ` Jesper Dangaard Brouer 2026-02-11 8:50 ` Paolo Abeni 2026-02-06 16:24 ` [PATCH net-next v3 2/6] net: sched: sfq: convert to qdisc drop reasons Jesper Dangaard Brouer 2026-02-10 11:48 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 3/6] net: sched: sch_cake: use enum qdisc_drop_reason for cobalt_should_drop Jesper Dangaard Brouer 2026-02-10 11:52 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 4/6] net: sched: rename QDISC_DROP_CAKE_FLOOD to QDISC_DROP_FLOOD_PROTECTION Jesper Dangaard Brouer 2026-02-10 11:53 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 5/6] net: sched: rename QDISC_DROP_FQ_* to generic names Jesper Dangaard Brouer 2026-02-10 11:54 ` Toke Høiland-Jørgensen 2026-02-06 16:24 ` [PATCH net-next v3 6/6] net: sched: sch_dualpi2: use qdisc_dequeue_drop() for dequeue drops Jesper Dangaard Brouer 2026-02-10 12:00 ` Toke Høiland-Jørgensen
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox