1 /*
2 * Copyright (c) 2018, Mellanox Technologies. All rights reserved.
3 *
4 * This software is available to you under a choice of one of two
5 * licenses. You may choose to be licensed under the terms of the GNU
6 * General Public License (GPL) Version 2, available from the file
7 * COPYING in the main directory of this source tree, or the
8 * OpenIB.org BSD license below:
9 *
10 * Redistribution and use in source and binary forms, with or
11 * without modification, are permitted provided that the following
12 * conditions are met:
13 *
14 * - Redistributions of source code must retain the above
15 * copyright notice, this list of conditions and the following
16 * disclaimer.
17 *
18 * - Redistributions in binary form must reproduce the above
19 * copyright notice, this list of conditions and the following
20 * disclaimer in the documentation and/or other materials
21 * provided with the distribution.
22 *
23 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
24 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
25 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
26 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
27 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
28 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
29 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
30 * SOFTWARE.
31 */
32
33 #ifndef _MLX5_FS_HELPERS_
34 #define _MLX5_FS_HELPERS_
35
36 #include <linux/mlx5/mlx5_ifc.h>
37
38 #define MLX5_FS_IPV4_VERSION 4
39 #define MLX5_FS_IPV6_VERSION 6
40
mlx5_fs_is_ipsec_flow(const u32 * match_c)41 static inline bool mlx5_fs_is_ipsec_flow(const u32 *match_c)
42 {
43 void *misc_params_c = MLX5_ADDR_OF(fte_match_param, match_c,
44 misc_parameters);
45
46 return MLX5_GET(fte_match_set_misc, misc_params_c, outer_esp_spi);
47 }
48
_mlx5_fs_is_outer_ipproto_flow(const u32 * match_c,const u32 * match_v,u8 match)49 static inline bool _mlx5_fs_is_outer_ipproto_flow(const u32 *match_c,
50 const u32 *match_v, u8 match)
51 {
52 const void *headers_c = MLX5_ADDR_OF(fte_match_param, match_c,
53 outer_headers);
54 const void *headers_v = MLX5_ADDR_OF(fte_match_param, match_v,
55 outer_headers);
56
57 return MLX5_GET(fte_match_set_lyr_2_4, headers_c, ip_protocol) == 0xff &&
58 MLX5_GET(fte_match_set_lyr_2_4, headers_v, ip_protocol) == match;
59 }
60
mlx5_fs_is_outer_tcp_flow(const u32 * match_c,const u32 * match_v)61 static inline bool mlx5_fs_is_outer_tcp_flow(const u32 *match_c,
62 const u32 *match_v)
63 {
64 return _mlx5_fs_is_outer_ipproto_flow(match_c, match_v, IPPROTO_TCP);
65 }
66
mlx5_fs_is_outer_udp_flow(const u32 * match_c,const u32 * match_v)67 static inline bool mlx5_fs_is_outer_udp_flow(const u32 *match_c,
68 const u32 *match_v)
69 {
70 return _mlx5_fs_is_outer_ipproto_flow(match_c, match_v, IPPROTO_UDP);
71 }
72
mlx5_fs_is_vxlan_flow(const u32 * match_c)73 static inline bool mlx5_fs_is_vxlan_flow(const u32 *match_c)
74 {
75 void *misc_params_c = MLX5_ADDR_OF(fte_match_param, match_c,
76 misc_parameters);
77
78 return MLX5_GET(fte_match_set_misc, misc_params_c, vxlan_vni);
79 }
80
_mlx5_fs_is_outer_ipv_flow(struct mlx5_core_dev * mdev,const u32 * match_c,const u32 * match_v,int version)81 static inline bool _mlx5_fs_is_outer_ipv_flow(struct mlx5_core_dev *mdev,
82 const u32 *match_c,
83 const u32 *match_v, int version)
84 {
85 int match_ipv = MLX5_CAP_FLOWTABLE_NIC_RX(mdev,
86 ft_field_support.outer_ip_version);
87 const void *headers_c = MLX5_ADDR_OF(fte_match_param, match_c,
88 outer_headers);
89 const void *headers_v = MLX5_ADDR_OF(fte_match_param, match_v,
90 outer_headers);
91
92 if (!match_ipv) {
93 u16 ethertype;
94
95 switch (version) {
96 case MLX5_FS_IPV4_VERSION:
97 ethertype = ETH_P_IP;
98 break;
99 case MLX5_FS_IPV6_VERSION:
100 ethertype = ETH_P_IPV6;
101 break;
102 default:
103 return false;
104 }
105
106 return MLX5_GET(fte_match_set_lyr_2_4, headers_c,
107 ethertype) == 0xffff &&
108 MLX5_GET(fte_match_set_lyr_2_4, headers_v,
109 ethertype) == ethertype;
110 }
111
112 return MLX5_GET(fte_match_set_lyr_2_4, headers_c,
113 ip_version) == 0xf &&
114 MLX5_GET(fte_match_set_lyr_2_4, headers_v,
115 ip_version) == version;
116 }
117
118 static inline bool
mlx5_fs_is_outer_ipv4_flow(struct mlx5_core_dev * mdev,const u32 * match_c,const u32 * match_v)119 mlx5_fs_is_outer_ipv4_flow(struct mlx5_core_dev *mdev, const u32 *match_c,
120 const u32 *match_v)
121 {
122 return _mlx5_fs_is_outer_ipv_flow(mdev, match_c, match_v,
123 MLX5_FS_IPV4_VERSION);
124 }
125
126 static inline bool
mlx5_fs_is_outer_ipv6_flow(struct mlx5_core_dev * mdev,const u32 * match_c,const u32 * match_v)127 mlx5_fs_is_outer_ipv6_flow(struct mlx5_core_dev *mdev, const u32 *match_c,
128 const u32 *match_v)
129 {
130 return _mlx5_fs_is_outer_ipv_flow(mdev, match_c, match_v,
131 MLX5_FS_IPV6_VERSION);
132 }
133
mlx5_fs_is_outer_ipsec_flow(const u32 * match_c)134 static inline bool mlx5_fs_is_outer_ipsec_flow(const u32 *match_c)
135 {
136 void *misc_params_c =
137 MLX5_ADDR_OF(fte_match_param, match_c, misc_parameters);
138
139 return MLX5_GET(fte_match_set_misc, misc_params_c, outer_esp_spi);
140 }
141
142 #endif
143