1*4882a593Smuzhiyun /* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */
2*4882a593Smuzhiyun /* Copyright (C) 2018 Netronome Systems, Inc. */
3*4882a593Smuzhiyun
4*4882a593Smuzhiyun #ifndef __NFP_ABM_H__
5*4882a593Smuzhiyun #define __NFP_ABM_H__ 1
6*4882a593Smuzhiyun
7*4882a593Smuzhiyun #include <linux/bits.h>
8*4882a593Smuzhiyun #include <linux/list.h>
9*4882a593Smuzhiyun #include <linux/radix-tree.h>
10*4882a593Smuzhiyun #include <net/devlink.h>
11*4882a593Smuzhiyun #include <net/pkt_cls.h>
12*4882a593Smuzhiyun #include <net/pkt_sched.h>
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun /* Dump of 64 PRIOs and 256 REDs seems to take 850us on Xeon v4 @ 2.20GHz;
15*4882a593Smuzhiyun * 2.5ms / 400Hz seems more than sufficient for stats resolution.
16*4882a593Smuzhiyun */
17*4882a593Smuzhiyun #define NFP_ABM_STATS_REFRESH_IVAL (2500 * 1000) /* ns */
18*4882a593Smuzhiyun
19*4882a593Smuzhiyun #define NFP_ABM_LVL_INFINITY S32_MAX
20*4882a593Smuzhiyun
21*4882a593Smuzhiyun struct nfp_app;
22*4882a593Smuzhiyun struct nfp_net;
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #define NFP_ABM_PORTID_TYPE GENMASK(23, 16)
25*4882a593Smuzhiyun #define NFP_ABM_PORTID_ID GENMASK(7, 0)
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun /* The possible actions if thresholds are exceeded */
28*4882a593Smuzhiyun enum nfp_abm_q_action {
29*4882a593Smuzhiyun /* mark if ECN capable, otherwise drop */
30*4882a593Smuzhiyun NFP_ABM_ACT_MARK_DROP = 0,
31*4882a593Smuzhiyun /* mark if ECN capable, otherwise goto QM */
32*4882a593Smuzhiyun NFP_ABM_ACT_MARK_QUEUE = 1,
33*4882a593Smuzhiyun NFP_ABM_ACT_DROP = 2,
34*4882a593Smuzhiyun NFP_ABM_ACT_QUEUE = 3,
35*4882a593Smuzhiyun NFP_ABM_ACT_NOQUEUE = 4,
36*4882a593Smuzhiyun };
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun /**
39*4882a593Smuzhiyun * struct nfp_abm - ABM NIC app structure
40*4882a593Smuzhiyun * @app: back pointer to nfp_app
41*4882a593Smuzhiyun * @pf_id: ID of our PF link
42*4882a593Smuzhiyun *
43*4882a593Smuzhiyun * @red_support: is RED offload supported
44*4882a593Smuzhiyun * @num_prios: number of supported DSCP priorities
45*4882a593Smuzhiyun * @num_bands: number of supported DSCP priority bands
46*4882a593Smuzhiyun * @action_mask: bitmask of supported actions
47*4882a593Smuzhiyun *
48*4882a593Smuzhiyun * @thresholds: current threshold configuration
49*4882a593Smuzhiyun * @threshold_undef: bitmap of thresholds which have not been set
50*4882a593Smuzhiyun * @actions: current FW action configuration
51*4882a593Smuzhiyun * @num_thresholds: number of @thresholds and bits in @threshold_undef
52*4882a593Smuzhiyun *
53*4882a593Smuzhiyun * @prio_map_len: computed length of FW priority map (in bytes)
54*4882a593Smuzhiyun * @dscp_mask: mask FW will apply on DSCP field
55*4882a593Smuzhiyun *
56*4882a593Smuzhiyun * @eswitch_mode: devlink eswitch mode, advanced functions only visible
57*4882a593Smuzhiyun * in switchdev mode
58*4882a593Smuzhiyun *
59*4882a593Smuzhiyun * @q_lvls: queue level control area
60*4882a593Smuzhiyun * @qm_stats: queue statistics symbol
61*4882a593Smuzhiyun * @q_stats: basic queue statistics (only in per-band case)
62*4882a593Smuzhiyun */
63*4882a593Smuzhiyun struct nfp_abm {
64*4882a593Smuzhiyun struct nfp_app *app;
65*4882a593Smuzhiyun unsigned int pf_id;
66*4882a593Smuzhiyun
67*4882a593Smuzhiyun unsigned int red_support;
68*4882a593Smuzhiyun unsigned int num_prios;
69*4882a593Smuzhiyun unsigned int num_bands;
70*4882a593Smuzhiyun unsigned int action_mask;
71*4882a593Smuzhiyun
72*4882a593Smuzhiyun u32 *thresholds;
73*4882a593Smuzhiyun unsigned long *threshold_undef;
74*4882a593Smuzhiyun u8 *actions;
75*4882a593Smuzhiyun size_t num_thresholds;
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun unsigned int prio_map_len;
78*4882a593Smuzhiyun u8 dscp_mask;
79*4882a593Smuzhiyun
80*4882a593Smuzhiyun enum devlink_eswitch_mode eswitch_mode;
81*4882a593Smuzhiyun
82*4882a593Smuzhiyun const struct nfp_rtsym *q_lvls;
83*4882a593Smuzhiyun const struct nfp_rtsym *qm_stats;
84*4882a593Smuzhiyun const struct nfp_rtsym *q_stats;
85*4882a593Smuzhiyun };
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun /**
88*4882a593Smuzhiyun * struct nfp_alink_stats - ABM NIC statistics
89*4882a593Smuzhiyun * @tx_pkts: number of TXed packets
90*4882a593Smuzhiyun * @tx_bytes: number of TXed bytes
91*4882a593Smuzhiyun * @backlog_pkts: momentary backlog length (packets)
92*4882a593Smuzhiyun * @backlog_bytes: momentary backlog length (bytes)
93*4882a593Smuzhiyun * @overlimits: number of ECN marked TXed packets (accumulative)
94*4882a593Smuzhiyun * @drops: number of tail-dropped packets (accumulative)
95*4882a593Smuzhiyun */
96*4882a593Smuzhiyun struct nfp_alink_stats {
97*4882a593Smuzhiyun u64 tx_pkts;
98*4882a593Smuzhiyun u64 tx_bytes;
99*4882a593Smuzhiyun u64 backlog_pkts;
100*4882a593Smuzhiyun u64 backlog_bytes;
101*4882a593Smuzhiyun u64 overlimits;
102*4882a593Smuzhiyun u64 drops;
103*4882a593Smuzhiyun };
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun /**
106*4882a593Smuzhiyun * struct nfp_alink_xstats - extended ABM NIC statistics
107*4882a593Smuzhiyun * @ecn_marked: number of ECN marked TXed packets
108*4882a593Smuzhiyun * @pdrop: number of hard drops due to queue limit
109*4882a593Smuzhiyun */
110*4882a593Smuzhiyun struct nfp_alink_xstats {
111*4882a593Smuzhiyun u64 ecn_marked;
112*4882a593Smuzhiyun u64 pdrop;
113*4882a593Smuzhiyun };
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun enum nfp_qdisc_type {
116*4882a593Smuzhiyun NFP_QDISC_NONE = 0,
117*4882a593Smuzhiyun NFP_QDISC_MQ,
118*4882a593Smuzhiyun NFP_QDISC_RED,
119*4882a593Smuzhiyun NFP_QDISC_GRED,
120*4882a593Smuzhiyun };
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun #define NFP_QDISC_UNTRACKED ((struct nfp_qdisc *)1UL)
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun /**
125*4882a593Smuzhiyun * struct nfp_qdisc - tracked TC Qdisc
126*4882a593Smuzhiyun * @netdev: netdev on which Qdisc was created
127*4882a593Smuzhiyun * @type: Qdisc type
128*4882a593Smuzhiyun * @handle: handle of this Qdisc
129*4882a593Smuzhiyun * @parent_handle: handle of the parent (unreliable if Qdisc was grafted)
130*4882a593Smuzhiyun * @use_cnt: number of attachment points in the hierarchy
131*4882a593Smuzhiyun * @num_children: current size of the @children array
132*4882a593Smuzhiyun * @children: pointers to children
133*4882a593Smuzhiyun *
134*4882a593Smuzhiyun * @params_ok: parameters of this Qdisc are OK for offload
135*4882a593Smuzhiyun * @offload_mark: offload refresh state - selected for offload
136*4882a593Smuzhiyun * @offloaded: Qdisc is currently offloaded to the HW
137*4882a593Smuzhiyun *
138*4882a593Smuzhiyun * @mq: MQ Qdisc specific parameters and state
139*4882a593Smuzhiyun * @mq.stats: current stats of the MQ Qdisc
140*4882a593Smuzhiyun * @mq.prev_stats: previously reported @mq.stats
141*4882a593Smuzhiyun *
142*4882a593Smuzhiyun * @red: RED Qdisc specific parameters and state
143*4882a593Smuzhiyun * @red.num_bands: Number of valid entries in the @red.band table
144*4882a593Smuzhiyun * @red.band: Per-band array of RED instances
145*4882a593Smuzhiyun * @red.band.ecn: ECN marking is enabled (rather than drop)
146*4882a593Smuzhiyun * @red.band.threshold: ECN marking threshold
147*4882a593Smuzhiyun * @red.band.stats: current stats of the RED Qdisc
148*4882a593Smuzhiyun * @red.band.prev_stats: previously reported @red.stats
149*4882a593Smuzhiyun * @red.band.xstats: extended stats for RED - current
150*4882a593Smuzhiyun * @red.band.prev_xstats: extended stats for RED - previously reported
151*4882a593Smuzhiyun */
152*4882a593Smuzhiyun struct nfp_qdisc {
153*4882a593Smuzhiyun struct net_device *netdev;
154*4882a593Smuzhiyun enum nfp_qdisc_type type;
155*4882a593Smuzhiyun u32 handle;
156*4882a593Smuzhiyun u32 parent_handle;
157*4882a593Smuzhiyun unsigned int use_cnt;
158*4882a593Smuzhiyun unsigned int num_children;
159*4882a593Smuzhiyun struct nfp_qdisc **children;
160*4882a593Smuzhiyun
161*4882a593Smuzhiyun bool params_ok;
162*4882a593Smuzhiyun bool offload_mark;
163*4882a593Smuzhiyun bool offloaded;
164*4882a593Smuzhiyun
165*4882a593Smuzhiyun union {
166*4882a593Smuzhiyun /* NFP_QDISC_MQ */
167*4882a593Smuzhiyun struct {
168*4882a593Smuzhiyun struct nfp_alink_stats stats;
169*4882a593Smuzhiyun struct nfp_alink_stats prev_stats;
170*4882a593Smuzhiyun } mq;
171*4882a593Smuzhiyun /* TC_SETUP_QDISC_RED, TC_SETUP_QDISC_GRED */
172*4882a593Smuzhiyun struct {
173*4882a593Smuzhiyun unsigned int num_bands;
174*4882a593Smuzhiyun
175*4882a593Smuzhiyun struct {
176*4882a593Smuzhiyun bool ecn;
177*4882a593Smuzhiyun u32 threshold;
178*4882a593Smuzhiyun struct nfp_alink_stats stats;
179*4882a593Smuzhiyun struct nfp_alink_stats prev_stats;
180*4882a593Smuzhiyun struct nfp_alink_xstats xstats;
181*4882a593Smuzhiyun struct nfp_alink_xstats prev_xstats;
182*4882a593Smuzhiyun } band[MAX_DPs];
183*4882a593Smuzhiyun } red;
184*4882a593Smuzhiyun };
185*4882a593Smuzhiyun };
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun /**
188*4882a593Smuzhiyun * struct nfp_abm_link - port tuple of a ABM NIC
189*4882a593Smuzhiyun * @abm: back pointer to nfp_abm
190*4882a593Smuzhiyun * @vnic: data vNIC
191*4882a593Smuzhiyun * @id: id of the data vNIC
192*4882a593Smuzhiyun * @queue_base: id of base to host queue within PCIe (not QC idx)
193*4882a593Smuzhiyun * @total_queues: number of PF queues
194*4882a593Smuzhiyun *
195*4882a593Smuzhiyun * @last_stats_update: ktime of last stats update
196*4882a593Smuzhiyun *
197*4882a593Smuzhiyun * @prio_map: current map of priorities
198*4882a593Smuzhiyun * @has_prio: @prio_map is valid
199*4882a593Smuzhiyun *
200*4882a593Smuzhiyun * @def_band: default band to use
201*4882a593Smuzhiyun * @dscp_map: list of DSCP to band mappings
202*4882a593Smuzhiyun *
203*4882a593Smuzhiyun * @root_qdisc: pointer to the current root of the Qdisc hierarchy
204*4882a593Smuzhiyun * @qdiscs: all qdiscs recorded by major part of the handle
205*4882a593Smuzhiyun */
206*4882a593Smuzhiyun struct nfp_abm_link {
207*4882a593Smuzhiyun struct nfp_abm *abm;
208*4882a593Smuzhiyun struct nfp_net *vnic;
209*4882a593Smuzhiyun unsigned int id;
210*4882a593Smuzhiyun unsigned int queue_base;
211*4882a593Smuzhiyun unsigned int total_queues;
212*4882a593Smuzhiyun
213*4882a593Smuzhiyun u64 last_stats_update;
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun u32 *prio_map;
216*4882a593Smuzhiyun bool has_prio;
217*4882a593Smuzhiyun
218*4882a593Smuzhiyun u8 def_band;
219*4882a593Smuzhiyun struct list_head dscp_map;
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun struct nfp_qdisc *root_qdisc;
222*4882a593Smuzhiyun struct radix_tree_root qdiscs;
223*4882a593Smuzhiyun };
224*4882a593Smuzhiyun
nfp_abm_has_prio(struct nfp_abm * abm)225*4882a593Smuzhiyun static inline bool nfp_abm_has_prio(struct nfp_abm *abm)
226*4882a593Smuzhiyun {
227*4882a593Smuzhiyun return abm->num_bands > 1;
228*4882a593Smuzhiyun }
229*4882a593Smuzhiyun
nfp_abm_has_drop(struct nfp_abm * abm)230*4882a593Smuzhiyun static inline bool nfp_abm_has_drop(struct nfp_abm *abm)
231*4882a593Smuzhiyun {
232*4882a593Smuzhiyun return abm->action_mask & BIT(NFP_ABM_ACT_DROP);
233*4882a593Smuzhiyun }
234*4882a593Smuzhiyun
nfp_abm_has_mark(struct nfp_abm * abm)235*4882a593Smuzhiyun static inline bool nfp_abm_has_mark(struct nfp_abm *abm)
236*4882a593Smuzhiyun {
237*4882a593Smuzhiyun return abm->action_mask & BIT(NFP_ABM_ACT_MARK_DROP);
238*4882a593Smuzhiyun }
239*4882a593Smuzhiyun
240*4882a593Smuzhiyun void nfp_abm_qdisc_offload_update(struct nfp_abm_link *alink);
241*4882a593Smuzhiyun int nfp_abm_setup_root(struct net_device *netdev, struct nfp_abm_link *alink,
242*4882a593Smuzhiyun struct tc_root_qopt_offload *opt);
243*4882a593Smuzhiyun int nfp_abm_setup_tc_red(struct net_device *netdev, struct nfp_abm_link *alink,
244*4882a593Smuzhiyun struct tc_red_qopt_offload *opt);
245*4882a593Smuzhiyun int nfp_abm_setup_tc_mq(struct net_device *netdev, struct nfp_abm_link *alink,
246*4882a593Smuzhiyun struct tc_mq_qopt_offload *opt);
247*4882a593Smuzhiyun int nfp_abm_setup_tc_gred(struct net_device *netdev, struct nfp_abm_link *alink,
248*4882a593Smuzhiyun struct tc_gred_qopt_offload *opt);
249*4882a593Smuzhiyun int nfp_abm_setup_cls_block(struct net_device *netdev, struct nfp_repr *repr,
250*4882a593Smuzhiyun struct flow_block_offload *opt);
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun int nfp_abm_ctrl_read_params(struct nfp_abm_link *alink);
253*4882a593Smuzhiyun int nfp_abm_ctrl_find_addrs(struct nfp_abm *abm);
254*4882a593Smuzhiyun int __nfp_abm_ctrl_set_q_lvl(struct nfp_abm *abm, unsigned int id, u32 val);
255*4882a593Smuzhiyun int nfp_abm_ctrl_set_q_lvl(struct nfp_abm_link *alink, unsigned int band,
256*4882a593Smuzhiyun unsigned int queue, u32 val);
257*4882a593Smuzhiyun int __nfp_abm_ctrl_set_q_act(struct nfp_abm *abm, unsigned int id,
258*4882a593Smuzhiyun enum nfp_abm_q_action act);
259*4882a593Smuzhiyun int nfp_abm_ctrl_set_q_act(struct nfp_abm_link *alink, unsigned int band,
260*4882a593Smuzhiyun unsigned int queue, enum nfp_abm_q_action act);
261*4882a593Smuzhiyun int nfp_abm_ctrl_read_q_stats(struct nfp_abm_link *alink,
262*4882a593Smuzhiyun unsigned int band, unsigned int queue,
263*4882a593Smuzhiyun struct nfp_alink_stats *stats);
264*4882a593Smuzhiyun int nfp_abm_ctrl_read_q_xstats(struct nfp_abm_link *alink,
265*4882a593Smuzhiyun unsigned int band, unsigned int queue,
266*4882a593Smuzhiyun struct nfp_alink_xstats *xstats);
267*4882a593Smuzhiyun u64 nfp_abm_ctrl_stat_non_sto(struct nfp_abm_link *alink, unsigned int i);
268*4882a593Smuzhiyun u64 nfp_abm_ctrl_stat_sto(struct nfp_abm_link *alink, unsigned int i);
269*4882a593Smuzhiyun int nfp_abm_ctrl_qm_enable(struct nfp_abm *abm);
270*4882a593Smuzhiyun int nfp_abm_ctrl_qm_disable(struct nfp_abm *abm);
271*4882a593Smuzhiyun void nfp_abm_prio_map_update(struct nfp_abm *abm);
272*4882a593Smuzhiyun int nfp_abm_ctrl_prio_map_update(struct nfp_abm_link *alink, u32 *packed);
273*4882a593Smuzhiyun #endif
274