xref: /dpdk/drivers/net/sfc/sfc_filter.c (revision 98d26ef7b839e43b0cdbdee216c7caca6b76f8f8)
1 /* SPDX-License-Identifier: BSD-3-Clause
2  *
3  * Copyright(c) 2019-2021 Xilinx, Inc.
4  * Copyright(c) 2017-2019 Solarflare Communications Inc.
5  *
6  * This software was jointly developed between OKTET Labs (under contract
7  * for Solarflare) and Solarflare Communications, Inc.
8  */
9 
10 #include <rte_common.h>
11 
12 #include "efx.h"
13 
14 #include "sfc.h"
15 #include "sfc_debug.h"
16 #include "sfc_log.h"
17 
18 boolean_t
sfc_filter_is_match_supported(struct sfc_adapter * sa,uint32_t match)19 sfc_filter_is_match_supported(struct sfc_adapter *sa, uint32_t match)
20 {
21 	struct sfc_filter *filter = &sa->filter;
22 	size_t i;
23 
24 	for (i = 0; i < filter->supported_match_num; ++i) {
25 		if (match == filter->supported_match[i])
26 			return B_TRUE;
27 	}
28 
29 	return B_FALSE;
30 }
31 
32 static int
sfc_filter_cache_match_supported(struct sfc_adapter * sa)33 sfc_filter_cache_match_supported(struct sfc_adapter *sa)
34 {
35 	struct sfc_filter *filter = &sa->filter;
36 	size_t num = filter->supported_match_num;
37 	uint32_t *buf = filter->supported_match;
38 	unsigned int retry;
39 	int rc;
40 
41 	/* Just a guess of possibly sufficient entries */
42 	if (num == 0)
43 		num = 16;
44 
45 	for (retry = 0; retry < 2; ++retry) {
46 		if (num != filter->supported_match_num) {
47 			rc = ENOMEM;
48 			buf = rte_realloc(buf, num * sizeof(*buf), 0);
49 			if (buf == NULL)
50 				goto fail_realloc;
51 		}
52 
53 		rc = efx_filter_supported_filters(sa->nic, buf, num, &num);
54 		if (rc == 0) {
55 			filter->supported_match_num = num;
56 			filter->supported_match = buf;
57 
58 			return 0;
59 		} else if (rc != ENOSPC) {
60 			goto fail_efx_filter_supported_filters;
61 		}
62 	}
63 
64 	SFC_ASSERT(rc == ENOSPC);
65 
66 fail_efx_filter_supported_filters:
67 fail_realloc:
68 	/* Original pointer is not freed by rte_realloc() on failure */
69 	rte_free(buf);
70 	filter->supported_match = NULL;
71 	filter->supported_match_num = 0;
72 	return rc;
73 }
74 
75 int
sfc_filter_attach(struct sfc_adapter * sa)76 sfc_filter_attach(struct sfc_adapter *sa)
77 {
78 	int rc;
79 	unsigned int i;
80 
81 	sfc_log_init(sa, "entry");
82 
83 	rc = efx_filter_init(sa->nic);
84 	if (rc != 0)
85 		goto fail_filter_init;
86 
87 	rc = sfc_filter_cache_match_supported(sa);
88 	if (rc != 0)
89 		goto fail_cache_match_supported;
90 
91 	efx_filter_fini(sa->nic);
92 
93 	sa->filter.supports_ip_proto_or_addr_filter = B_FALSE;
94 	sa->filter.supports_rem_or_local_port_filter = B_FALSE;
95 	for (i = 0; i < sa->filter.supported_match_num; ++i) {
96 		if (sa->filter.supported_match[i] &
97 		    (EFX_FILTER_MATCH_IP_PROTO | EFX_FILTER_MATCH_LOC_HOST |
98 		     EFX_FILTER_MATCH_REM_HOST))
99 			sa->filter.supports_ip_proto_or_addr_filter = B_TRUE;
100 
101 		if (sa->filter.supported_match[i] &
102 		    (EFX_FILTER_MATCH_LOC_PORT | EFX_FILTER_MATCH_REM_PORT))
103 			sa->filter.supports_rem_or_local_port_filter = B_TRUE;
104 	}
105 
106 	sfc_log_init(sa, "done");
107 
108 	return 0;
109 
110 fail_cache_match_supported:
111 	efx_filter_fini(sa->nic);
112 
113 fail_filter_init:
114 	sfc_log_init(sa, "failed %d", rc);
115 	return rc;
116 }
117 
118 void
sfc_filter_detach(struct sfc_adapter * sa)119 sfc_filter_detach(struct sfc_adapter *sa)
120 {
121 	struct sfc_filter *filter = &sa->filter;
122 
123 	sfc_log_init(sa, "entry");
124 
125 	rte_free(filter->supported_match);
126 	filter->supported_match = NULL;
127 	filter->supported_match_num = 0;
128 
129 	sfc_log_init(sa, "done");
130 }
131