1 // SPDX-License-Identifier: GPL-2.0
2 /* Marvell OcteonTx2 RVU Admin Function driver
4 * Copyright (C) 2018 Marvell International Ltd.
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
11 #include <linux/module.h>
12 #include <linux/pci.h>
14 #include "rvu_struct.h"
18 #include "npc_profile.h"
20 #define RSVD_MCAM_ENTRIES_PER_PF 2 /* Bcast & Promisc */
21 #define RSVD_MCAM_ENTRIES_PER_NIXLF 1 /* Ucast for LFs */
23 #define NIXLF_UCAST_ENTRY 0
24 #define NIXLF_BCAST_ENTRY 1
25 #define NIXLF_PROMISC_ENTRY 2
27 #define NPC_PARSE_RESULT_DMAC_OFFSET 8
30 #define NPC_MAX_KWS_IN_KEY 7 /* Number of keywords in max keywidth */
31 u64 kw[NPC_MAX_KWS_IN_KEY];
32 u64 kw_mask[NPC_MAX_KWS_IN_KEY];
37 void rvu_npc_set_pkind(struct rvu *rvu, int pkind, struct rvu_pfvf *pfvf)
42 blkaddr = rvu_get_blkaddr(rvu, BLKTYPE_NPC, 0);
46 /* Config CPI base for the PKIND */
47 val = pkind | 1ULL << 62;
48 rvu_write64(rvu, blkaddr, NPC_AF_PKINDX_CPI_DEFX(pkind, 0), val);
51 int rvu_npc_get_pkind(struct rvu *rvu, u16 pf)
53 struct npc_pkind *pkind = &rvu->hw->pkind;
57 for (i = 0; i < pkind->rsrc.max; i++) {
58 map = pkind->pfchan_map[i];
59 if (((map >> 16) & 0x3F) == pf)
65 static int npc_get_nixlf_mcam_index(struct npc_mcam *mcam,
66 u16 pcifunc, int nixlf, int type)
68 int pf = rvu_get_pf(pcifunc);
71 /* Check if this is for a PF */
72 if (pf && !(pcifunc & RVU_PFVF_FUNC_MASK)) {
73 /* Reserved entries exclude PF0 */
75 index = mcam->pf_offset + (pf * RSVD_MCAM_ENTRIES_PER_PF);
76 /* Broadcast address matching entry should be first so
77 * that the packet can be replicated to all VFs.
79 if (type == NIXLF_BCAST_ENTRY)
81 else if (type == NIXLF_PROMISC_ENTRY)
85 return (mcam->nixlf_offset + (nixlf * RSVD_MCAM_ENTRIES_PER_NIXLF));
88 static int npc_get_bank(struct npc_mcam *mcam, int index)
90 int bank = index / mcam->banksize;
92 /* 0,1 & 2,3 banks are combined for this keysize */
93 if (mcam->keysize == NPC_MCAM_KEY_X2)
99 static bool is_mcam_entry_enabled(struct rvu *rvu, struct npc_mcam *mcam,
100 int blkaddr, int index)
102 int bank = npc_get_bank(mcam, index);
105 index &= (mcam->banksize - 1);
106 cfg = rvu_read64(rvu, blkaddr, NPC_AF_MCAMEX_BANKX_CFG(index, bank));
110 static void npc_enable_mcam_entry(struct rvu *rvu, struct npc_mcam *mcam,
111 int blkaddr, int index, bool enable)
113 int bank = npc_get_bank(mcam, index);
116 index &= (mcam->banksize - 1);
117 for (; bank < (actbank + mcam->banks_per_entry); bank++) {
118 rvu_write64(rvu, blkaddr,
119 NPC_AF_MCAMEX_BANKX_CFG(index, bank),
124 static void npc_get_keyword(struct mcam_entry *entry, int idx,
125 u64 *cam0, u64 *cam1)
129 #define CAM_MASK(n) (BIT_ULL(n) - 1)
131 /* 0, 2, 4, 6 indices refer to BANKX_CAMX_W0 and
132 * 1, 3, 5, 7 indices refer to BANKX_CAMX_W1.
134 * Also, only 48 bits of BANKX_CAMX_W1 are valid.
138 /* BANK(X)_CAM_W0<63:0> = MCAM_KEY[KW0]<63:0> */
139 *cam1 = entry->kw[0];
140 kw_mask = entry->kw_mask[0];
143 /* BANK(X)_CAM_W1<47:0> = MCAM_KEY[KW1]<47:0> */
144 *cam1 = entry->kw[1] & CAM_MASK(48);
145 kw_mask = entry->kw_mask[1] & CAM_MASK(48);
148 /* BANK(X + 1)_CAM_W0<15:0> = MCAM_KEY[KW1]<63:48>
149 * BANK(X + 1)_CAM_W0<63:16> = MCAM_KEY[KW2]<47:0>
151 *cam1 = (entry->kw[1] >> 48) & CAM_MASK(16);
152 *cam1 |= ((entry->kw[2] & CAM_MASK(48)) << 16);
153 kw_mask = (entry->kw_mask[1] >> 48) & CAM_MASK(16);
154 kw_mask |= ((entry->kw_mask[2] & CAM_MASK(48)) << 16);
157 /* BANK(X + 1)_CAM_W1<15:0> = MCAM_KEY[KW2]<63:48>
158 * BANK(X + 1)_CAM_W1<47:16> = MCAM_KEY[KW3]<31:0>
160 *cam1 = (entry->kw[2] >> 48) & CAM_MASK(16);
161 *cam1 |= ((entry->kw[3] & CAM_MASK(32)) << 16);
162 kw_mask = (entry->kw_mask[2] >> 48) & CAM_MASK(16);
163 kw_mask |= ((entry->kw_mask[3] & CAM_MASK(32)) << 16);
166 /* BANK(X + 2)_CAM_W0<31:0> = MCAM_KEY[KW3]<63:32>
167 * BANK(X + 2)_CAM_W0<63:32> = MCAM_KEY[KW4]<31:0>
169 *cam1 = (entry->kw[3] >> 32) & CAM_MASK(32);
170 *cam1 |= ((entry->kw[4] & CAM_MASK(32)) << 32);
171 kw_mask = (entry->kw_mask[3] >> 32) & CAM_MASK(32);
172 kw_mask |= ((entry->kw_mask[4] & CAM_MASK(32)) << 32);
175 /* BANK(X + 2)_CAM_W1<31:0> = MCAM_KEY[KW4]<63:32>
176 * BANK(X + 2)_CAM_W1<47:32> = MCAM_KEY[KW5]<15:0>
178 *cam1 = (entry->kw[4] >> 32) & CAM_MASK(32);
179 *cam1 |= ((entry->kw[5] & CAM_MASK(16)) << 32);
180 kw_mask = (entry->kw_mask[4] >> 32) & CAM_MASK(32);
181 kw_mask |= ((entry->kw_mask[5] & CAM_MASK(16)) << 32);
184 /* BANK(X + 3)_CAM_W0<47:0> = MCAM_KEY[KW5]<63:16>
185 * BANK(X + 3)_CAM_W0<63:48> = MCAM_KEY[KW6]<15:0>
187 *cam1 = (entry->kw[5] >> 16) & CAM_MASK(48);
188 *cam1 |= ((entry->kw[6] & CAM_MASK(16)) << 48);
189 kw_mask = (entry->kw_mask[5] >> 16) & CAM_MASK(48);
190 kw_mask |= ((entry->kw_mask[6] & CAM_MASK(16)) << 48);
193 /* BANK(X + 3)_CAM_W1<47:0> = MCAM_KEY[KW6]<63:16> */
194 *cam1 = (entry->kw[6] >> 16) & CAM_MASK(48);
195 kw_mask = (entry->kw_mask[6] >> 16) & CAM_MASK(48);
200 *cam0 = ~*cam1 & kw_mask;
203 static void npc_config_mcam_entry(struct rvu *rvu, struct npc_mcam *mcam,
204 int blkaddr, int index, u8 intf,
205 struct mcam_entry *entry, bool enable)
207 int bank = npc_get_bank(mcam, index);
208 int kw = 0, actbank, actindex;
211 actbank = bank; /* Save bank id, to set action later on */
213 index &= (mcam->banksize - 1);
215 /* CAM1 takes the comparison value and
216 * CAM0 specifies match for a bit in key being '0' or '1' or 'dontcare'.
217 * CAM1<n> = 0 & CAM0<n> = 1 => match if key<n> = 0
218 * CAM1<n> = 1 & CAM0<n> = 0 => match if key<n> = 1
219 * CAM1<n> = 0 & CAM0<n> = 0 => always match i.e dontcare.
221 for (; bank < (actbank + mcam->banks_per_entry); bank++, kw = kw + 2) {
222 /* Interface should be set in all banks */
223 rvu_write64(rvu, blkaddr,
224 NPC_AF_MCAMEX_BANKX_CAMX_INTF(index, bank, 1),
226 rvu_write64(rvu, blkaddr,
227 NPC_AF_MCAMEX_BANKX_CAMX_INTF(index, bank, 0),
230 /* Set the match key */
231 npc_get_keyword(entry, kw, &cam0, &cam1);
232 rvu_write64(rvu, blkaddr,
233 NPC_AF_MCAMEX_BANKX_CAMX_W0(index, bank, 1), cam1);
234 rvu_write64(rvu, blkaddr,
235 NPC_AF_MCAMEX_BANKX_CAMX_W0(index, bank, 0), cam0);
237 npc_get_keyword(entry, kw + 1, &cam0, &cam1);
238 rvu_write64(rvu, blkaddr,
239 NPC_AF_MCAMEX_BANKX_CAMX_W1(index, bank, 1), cam1);
240 rvu_write64(rvu, blkaddr,
241 NPC_AF_MCAMEX_BANKX_CAMX_W1(index, bank, 0), cam0);
245 rvu_write64(rvu, blkaddr,
246 NPC_AF_MCAMEX_BANKX_ACTION(index, actbank), entry->action);
248 /* Set TAG 'action' */
249 rvu_write64(rvu, blkaddr, NPC_AF_MCAMEX_BANKX_TAG_ACT(index, actbank),
252 /* Enable the entry */
254 npc_enable_mcam_entry(rvu, mcam, blkaddr, actindex, true);
256 npc_enable_mcam_entry(rvu, mcam, blkaddr, actindex, false);
259 static u64 npc_get_mcam_action(struct rvu *rvu, struct npc_mcam *mcam,
260 int blkaddr, int index)
262 int bank = npc_get_bank(mcam, index);
264 index &= (mcam->banksize - 1);
265 return rvu_read64(rvu, blkaddr,
266 NPC_AF_MCAMEX_BANKX_ACTION(index, bank));
269 void rvu_npc_install_ucast_entry(struct rvu *rvu, u16 pcifunc,
270 int nixlf, u64 chan, u8 *mac_addr)
272 struct npc_mcam *mcam = &rvu->hw->mcam;
273 struct mcam_entry entry = { {0} };
274 struct nix_rx_action action;
275 int blkaddr, index, kwi;
278 blkaddr = rvu_get_blkaddr(rvu, BLKTYPE_NPC, 0);
282 for (index = ETH_ALEN - 1; index >= 0; index--)
283 mac |= ((u64)*mac_addr++) << (8 * index);
285 index = npc_get_nixlf_mcam_index(mcam, pcifunc,
286 nixlf, NIXLF_UCAST_ENTRY);
288 /* Match ingress channel and DMAC */
290 entry.kw_mask[0] = 0xFFFULL;
292 kwi = NPC_PARSE_RESULT_DMAC_OFFSET / sizeof(u64);
294 entry.kw_mask[kwi] = BIT_ULL(48) - 1;
296 /* Don't change the action if entry is already enabled
297 * Otherwise RSS action may get overwritten.
299 if (is_mcam_entry_enabled(rvu, mcam, blkaddr, index)) {
300 *(u64 *)&action = npc_get_mcam_action(rvu, mcam,
303 *(u64 *)&action = 0x00;
304 action.op = NIX_RX_ACTIONOP_UCAST;
305 action.pf_func = pcifunc;
308 entry.action = *(u64 *)&action;
309 npc_config_mcam_entry(rvu, mcam, blkaddr, index,
310 NIX_INTF_RX, &entry, true);
313 void rvu_npc_install_bcast_match_entry(struct rvu *rvu, u16 pcifunc,
316 struct npc_mcam *mcam = &rvu->hw->mcam;
317 struct mcam_entry entry = { {0} };
318 struct nix_rx_action action;
320 struct rvu_pfvf *pfvf;
324 blkaddr = rvu_get_blkaddr(rvu, BLKTYPE_NPC, 0);
328 /* Only PF can add a bcast match entry */
329 if (pcifunc & RVU_PFVF_FUNC_MASK)
332 pfvf = rvu_get_pfvf(rvu, pcifunc & ~RVU_PFVF_FUNC_MASK);
335 index = npc_get_nixlf_mcam_index(mcam, pcifunc,
336 nixlf, NIXLF_BCAST_ENTRY);
338 /* Check for L2B bit and LMAC channel */
339 entry.kw[0] = BIT_ULL(25) | chan;
340 entry.kw_mask[0] = BIT_ULL(25) | 0xFFFULL;
342 *(u64 *)&action = 0x00;
344 /* Early silicon doesn't support pkt replication,
345 * so install entry with UCAST action, so that PF
346 * receives all broadcast packets.
348 action.op = NIX_RX_ACTIONOP_MCAST;
349 action.pf_func = pcifunc;
350 action.index = pfvf->bcast_mce_idx;
352 action.op = NIX_RX_ACTIONOP_UCAST;
353 action.pf_func = pcifunc;
356 entry.action = *(u64 *)&action;
357 npc_config_mcam_entry(rvu, mcam, blkaddr, index,
358 NIX_INTF_RX, &entry, true);
361 void rvu_npc_disable_mcam_entries(struct rvu *rvu, u16 pcifunc, int nixlf)
363 struct npc_mcam *mcam = &rvu->hw->mcam;
364 struct nix_rx_action action;
365 int blkaddr, index, bank;
367 blkaddr = rvu_get_blkaddr(rvu, BLKTYPE_NPC, 0);
371 /* Disable ucast MCAM match entry of this PF/VF */
372 index = npc_get_nixlf_mcam_index(mcam, pcifunc,
373 nixlf, NIXLF_UCAST_ENTRY);
374 npc_enable_mcam_entry(rvu, mcam, blkaddr, index, false);
376 /* For PF, disable promisc and bcast MCAM match entries */
377 if (!(pcifunc & RVU_PFVF_FUNC_MASK)) {
378 index = npc_get_nixlf_mcam_index(mcam, pcifunc,
379 nixlf, NIXLF_BCAST_ENTRY);
380 /* For bcast, disable only if it's action is not
381 * packet replication, incase if action is replication
382 * then this PF's nixlf is removed from bcast replication
385 bank = npc_get_bank(mcam, index);
386 index &= (mcam->banksize - 1);
387 *(u64 *)&action = rvu_read64(rvu, blkaddr,
388 NPC_AF_MCAMEX_BANKX_ACTION(index, bank));
389 if (action.op != NIX_RX_ACTIONOP_MCAST)
390 npc_enable_mcam_entry(rvu, mcam, blkaddr, index, false);
394 #define LDATA_EXTRACT_CONFIG(intf, lid, ltype, ld, cfg) \
395 rvu_write64(rvu, blkaddr, \
396 NPC_AF_INTFX_LIDX_LTX_LDX_CFG(intf, lid, ltype, ld), cfg)
398 #define LDATA_FLAGS_CONFIG(intf, ld, flags, cfg) \
399 rvu_write64(rvu, blkaddr, \
400 NPC_AF_INTFX_LDATAX_FLAGSX_CFG(intf, ld, flags), cfg)
402 static void npc_config_ldata_extract(struct rvu *rvu, int blkaddr)
404 struct npc_mcam *mcam = &rvu->hw->mcam;
409 cfg = rvu_read64(rvu, blkaddr, NPC_AF_CONST);
410 lid_count = (cfg >> 4) & 0xF;
412 /* First clear any existing config i.e
413 * disable LDATA and FLAGS extraction.
415 for (lid = 0; lid < lid_count; lid++) {
416 for (ltype = 0; ltype < 16; ltype++) {
417 LDATA_EXTRACT_CONFIG(NIX_INTF_RX, lid, ltype, 0, 0ULL);
418 LDATA_EXTRACT_CONFIG(NIX_INTF_RX, lid, ltype, 1, 0ULL);
419 LDATA_EXTRACT_CONFIG(NIX_INTF_TX, lid, ltype, 0, 0ULL);
420 LDATA_EXTRACT_CONFIG(NIX_INTF_TX, lid, ltype, 1, 0ULL);
422 LDATA_FLAGS_CONFIG(NIX_INTF_RX, 0, ltype, 0ULL);
423 LDATA_FLAGS_CONFIG(NIX_INTF_RX, 1, ltype, 0ULL);
424 LDATA_FLAGS_CONFIG(NIX_INTF_TX, 0, ltype, 0ULL);
425 LDATA_FLAGS_CONFIG(NIX_INTF_TX, 1, ltype, 0ULL);
429 /* If we plan to extract Outer IPv4 tuple for TCP/UDP pkts
430 * then 112bit key is not sufficient
432 if (mcam->keysize != NPC_MCAM_KEY_X2)
435 /* Start placing extracted data/flags from 64bit onwards, for now */
436 /* Extract DMAC from the packet */
437 cfg = (0x05 << 16) | BIT_ULL(7) | NPC_PARSE_RESULT_DMAC_OFFSET;
438 LDATA_EXTRACT_CONFIG(NIX_INTF_RX, NPC_LID_LA, NPC_LT_LA_ETHER, 0, cfg);
441 static void npc_config_kpuaction(struct rvu *rvu, int blkaddr,
442 struct npc_kpu_profile_action *kpuaction,
443 int kpu, int entry, bool pkind)
445 struct npc_kpu_action0 action0 = {0};
446 struct npc_kpu_action1 action1 = {0};
449 action1.errlev = kpuaction->errlev;
450 action1.errcode = kpuaction->errcode;
451 action1.dp0_offset = kpuaction->dp0_offset;
452 action1.dp1_offset = kpuaction->dp1_offset;
453 action1.dp2_offset = kpuaction->dp2_offset;
456 reg = NPC_AF_PKINDX_ACTION1(entry);
458 reg = NPC_AF_KPUX_ENTRYX_ACTION1(kpu, entry);
460 rvu_write64(rvu, blkaddr, reg, *(u64 *)&action1);
462 action0.byp_count = kpuaction->bypass_count;
463 action0.capture_ena = kpuaction->cap_ena;
464 action0.parse_done = kpuaction->parse_done;
465 action0.next_state = kpuaction->next_state;
466 action0.capture_lid = kpuaction->lid;
467 action0.capture_ltype = kpuaction->ltype;
468 action0.capture_flags = kpuaction->flags;
469 action0.ptr_advance = kpuaction->ptr_advance;
470 action0.var_len_offset = kpuaction->offset;
471 action0.var_len_mask = kpuaction->mask;
472 action0.var_len_right = kpuaction->right;
473 action0.var_len_shift = kpuaction->shift;
476 reg = NPC_AF_PKINDX_ACTION0(entry);
478 reg = NPC_AF_KPUX_ENTRYX_ACTION0(kpu, entry);
480 rvu_write64(rvu, blkaddr, reg, *(u64 *)&action0);
483 static void npc_config_kpucam(struct rvu *rvu, int blkaddr,
484 struct npc_kpu_profile_cam *kpucam,
487 struct npc_kpu_cam cam0 = {0};
488 struct npc_kpu_cam cam1 = {0};
490 cam1.state = kpucam->state & kpucam->state_mask;
491 cam1.dp0_data = kpucam->dp0 & kpucam->dp0_mask;
492 cam1.dp1_data = kpucam->dp1 & kpucam->dp1_mask;
493 cam1.dp2_data = kpucam->dp2 & kpucam->dp2_mask;
495 cam0.state = ~kpucam->state & kpucam->state_mask;
496 cam0.dp0_data = ~kpucam->dp0 & kpucam->dp0_mask;
497 cam0.dp1_data = ~kpucam->dp1 & kpucam->dp1_mask;
498 cam0.dp2_data = ~kpucam->dp2 & kpucam->dp2_mask;
500 rvu_write64(rvu, blkaddr,
501 NPC_AF_KPUX_ENTRYX_CAMX(kpu, entry, 0), *(u64 *)&cam0);
502 rvu_write64(rvu, blkaddr,
503 NPC_AF_KPUX_ENTRYX_CAMX(kpu, entry, 1), *(u64 *)&cam1);
506 static inline u64 enable_mask(int count)
508 return (((count) < 64) ? ~(BIT_ULL(count) - 1) : (0x00ULL));
511 static void npc_program_kpu_profile(struct rvu *rvu, int blkaddr, int kpu,
512 struct npc_kpu_profile *profile)
514 int entry, num_entries, max_entries;
516 if (profile->cam_entries != profile->action_entries) {
518 "KPU%d: CAM and action entries [%d != %d] not equal\n",
519 kpu, profile->cam_entries, profile->action_entries);
522 max_entries = rvu_read64(rvu, blkaddr, NPC_AF_CONST1) & 0xFFF;
524 /* Program CAM match entries for previous KPU extracted data */
525 num_entries = min_t(int, profile->cam_entries, max_entries);
526 for (entry = 0; entry < num_entries; entry++)
527 npc_config_kpucam(rvu, blkaddr,
528 &profile->cam[entry], kpu, entry);
530 /* Program this KPU's actions */
531 num_entries = min_t(int, profile->action_entries, max_entries);
532 for (entry = 0; entry < num_entries; entry++)
533 npc_config_kpuaction(rvu, blkaddr, &profile->action[entry],
536 /* Enable all programmed entries */
537 num_entries = min_t(int, profile->action_entries, profile->cam_entries);
538 rvu_write64(rvu, blkaddr,
539 NPC_AF_KPUX_ENTRY_DISX(kpu, 0), enable_mask(num_entries));
540 if (num_entries > 64) {
541 rvu_write64(rvu, blkaddr,
542 NPC_AF_KPUX_ENTRY_DISX(kpu, 1),
543 enable_mask(num_entries - 64));
546 /* Enable this KPU */
547 rvu_write64(rvu, blkaddr, NPC_AF_KPUX_CFG(kpu), 0x01);
550 static void npc_parser_profile_init(struct rvu *rvu, int blkaddr)
552 struct rvu_hwinfo *hw = rvu->hw;
553 int num_pkinds, num_kpus, idx;
554 struct npc_pkind *pkind;
557 hw->npc_kpus = (rvu_read64(rvu, blkaddr, NPC_AF_CONST) >> 8) & 0x1F;
559 /* Disable all KPUs and their entries */
560 for (idx = 0; idx < hw->npc_kpus; idx++) {
561 rvu_write64(rvu, blkaddr,
562 NPC_AF_KPUX_ENTRY_DISX(idx, 0), ~0ULL);
563 rvu_write64(rvu, blkaddr,
564 NPC_AF_KPUX_ENTRY_DISX(idx, 1), ~0ULL);
565 rvu_write64(rvu, blkaddr, NPC_AF_KPUX_CFG(idx), 0x00);
568 /* First program IKPU profile i.e PKIND configs.
569 * Check HW max count to avoid configuring junk or
570 * writing to unsupported CSR addresses.
573 num_pkinds = ARRAY_SIZE(ikpu_action_entries);
574 num_pkinds = min_t(int, pkind->rsrc.max, num_pkinds);
576 for (idx = 0; idx < num_pkinds; idx++)
577 npc_config_kpuaction(rvu, blkaddr,
578 &ikpu_action_entries[idx], 0, idx, true);
580 /* Program KPU CAM and Action profiles */
581 num_kpus = ARRAY_SIZE(npc_kpu_profiles);
582 num_kpus = min_t(int, hw->npc_kpus, num_kpus);
584 for (idx = 0; idx < num_kpus; idx++)
585 npc_program_kpu_profile(rvu, blkaddr,
586 idx, &npc_kpu_profiles[idx]);
589 static int npc_mcam_rsrcs_init(struct rvu *rvu, int blkaddr)
591 int nixlf_count = rvu_get_nixlf_count(rvu);
592 struct npc_mcam *mcam = &rvu->hw->mcam;
597 cfg = rvu_read64(rvu, blkaddr, NPC_AF_CONST);
598 mcam->banks = (cfg >> 44) & 0xF;
599 mcam->banksize = (cfg >> 28) & 0xFFFF;
601 /* Actual number of MCAM entries vary by entry size */
602 cfg = (rvu_read64(rvu, blkaddr,
603 NPC_AF_INTFX_KEX_CFG(0)) >> 32) & 0x07;
604 mcam->total_entries = (mcam->banks / BIT_ULL(cfg)) * mcam->banksize;
607 /* Number of banks combined per MCAM entry */
608 if (cfg == NPC_MCAM_KEY_X4)
609 mcam->banks_per_entry = 4;
610 else if (cfg == NPC_MCAM_KEY_X2)
611 mcam->banks_per_entry = 2;
613 mcam->banks_per_entry = 1;
615 /* Reserve one MCAM entry for each of the NIX LF to
616 * guarantee space to install default matching DMAC rule.
617 * Also reserve 2 MCAM entries for each PF for default
618 * channel based matching or 'bcast & promisc' matching to
619 * support BCAST and PROMISC modes of operation for PFs.
622 rsvd = (nixlf_count * RSVD_MCAM_ENTRIES_PER_NIXLF) +
623 ((rvu->hw->total_pfs - 1) * RSVD_MCAM_ENTRIES_PER_PF);
624 if (mcam->total_entries <= rsvd) {
626 "Insufficient NPC MCAM size %d for pkt I/O, exiting\n",
627 mcam->total_entries);
631 mcam->entries = mcam->total_entries - rsvd;
632 mcam->nixlf_offset = mcam->entries;
633 mcam->pf_offset = mcam->nixlf_offset + nixlf_count;
635 spin_lock_init(&mcam->lock);
640 int rvu_npc_init(struct rvu *rvu)
642 struct npc_pkind *pkind = &rvu->hw->pkind;
643 u64 keyz = NPC_MCAM_KEY_X2;
646 blkaddr = rvu_get_blkaddr(rvu, BLKTYPE_NPC, 0);
648 dev_err(rvu->dev, "%s: NPC block not implemented\n", __func__);
652 /* Allocate resource bimap for pkind*/
653 pkind->rsrc.max = (rvu_read64(rvu, blkaddr,
654 NPC_AF_CONST1) >> 12) & 0xFF;
655 err = rvu_alloc_bitmap(&pkind->rsrc);
659 /* Allocate mem for pkind to PF and channel mapping info */
660 pkind->pfchan_map = devm_kcalloc(rvu->dev, pkind->rsrc.max,
661 sizeof(u32), GFP_KERNEL);
662 if (!pkind->pfchan_map)
665 /* Configure KPU profile */
666 npc_parser_profile_init(rvu, blkaddr);
668 /* Config Outer L2, IPv4's NPC layer info */
669 rvu_write64(rvu, blkaddr, NPC_AF_PCK_DEF_OL2,
670 (NPC_LID_LA << 8) | (NPC_LT_LA_ETHER << 4) | 0x0F);
671 rvu_write64(rvu, blkaddr, NPC_AF_PCK_DEF_OIP4,
672 (NPC_LID_LC << 8) | (NPC_LT_LC_IP << 4) | 0x0F);
674 /* Enable below for Rx pkts.
675 * - Outer IPv4 header checksum validation.
676 * - Detect outer L2 broadcast address and set NPC_RESULT_S[L2M].
678 rvu_write64(rvu, blkaddr, NPC_AF_PCK_CFG,
679 rvu_read64(rvu, blkaddr, NPC_AF_PCK_CFG) |
680 BIT_ULL(6) | BIT_ULL(2));
682 /* Set RX and TX side MCAM search key size.
683 * Also enable parse key extract nibbles suchthat except
684 * layer E to H, rest of the key is included for MCAM search.
686 rvu_write64(rvu, blkaddr, NPC_AF_INTFX_KEX_CFG(NIX_INTF_RX),
687 ((keyz & 0x3) << 32) | ((1ULL << 20) - 1));
688 rvu_write64(rvu, blkaddr, NPC_AF_INTFX_KEX_CFG(NIX_INTF_TX),
689 ((keyz & 0x3) << 32) | ((1ULL << 20) - 1));
691 err = npc_mcam_rsrcs_init(rvu, blkaddr);
695 /* Config packet data and flags extraction into PARSE result */
696 npc_config_ldata_extract(rvu, blkaddr);
698 /* Set TX miss action to UCAST_DEFAULT i.e
699 * transmit the packet on NIX LF SQ's default channel.
701 rvu_write64(rvu, blkaddr, NPC_AF_INTFX_MISS_ACT(NIX_INTF_TX),
702 NIX_TX_ACTIONOP_UCAST_DEFAULT);
704 /* If MCAM lookup doesn't result in a match, drop the received packet */
705 rvu_write64(rvu, blkaddr, NPC_AF_INTFX_MISS_ACT(NIX_INTF_RX),
706 NIX_RX_ACTIONOP_DROP);
711 void rvu_npc_freemem(struct rvu *rvu)
713 struct npc_pkind *pkind = &rvu->hw->pkind;
715 kfree(pkind->rsrc.bmap);