xref: /openbmc/linux/net/bridge/br_cfm_netlink.c (revision 36fe4655)
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 
3 #include <net/genetlink.h>
4 
5 #include "br_private.h"
6 #include "br_private_cfm.h"
7 
8 static const struct nla_policy
9 br_cfm_mep_create_policy[IFLA_BRIDGE_CFM_MEP_CREATE_MAX + 1] = {
10 	[IFLA_BRIDGE_CFM_MEP_CREATE_UNSPEC]	= { .type = NLA_REJECT },
11 	[IFLA_BRIDGE_CFM_MEP_CREATE_INSTANCE]	= { .type = NLA_U32 },
12 	[IFLA_BRIDGE_CFM_MEP_CREATE_DOMAIN]	= { .type = NLA_U32 },
13 	[IFLA_BRIDGE_CFM_MEP_CREATE_DIRECTION]	= { .type = NLA_U32 },
14 	[IFLA_BRIDGE_CFM_MEP_CREATE_IFINDEX]	= { .type = NLA_U32 },
15 };
16 
17 static const struct nla_policy
18 br_cfm_mep_delete_policy[IFLA_BRIDGE_CFM_MEP_DELETE_MAX + 1] = {
19 	[IFLA_BRIDGE_CFM_MEP_DELETE_UNSPEC]	= { .type = NLA_REJECT },
20 	[IFLA_BRIDGE_CFM_MEP_DELETE_INSTANCE]	= { .type = NLA_U32 },
21 };
22 
23 static const struct nla_policy
24 br_cfm_mep_config_policy[IFLA_BRIDGE_CFM_MEP_CONFIG_MAX + 1] = {
25 	[IFLA_BRIDGE_CFM_MEP_CONFIG_UNSPEC]	 = { .type = NLA_REJECT },
26 	[IFLA_BRIDGE_CFM_MEP_CONFIG_INSTANCE]	 = { .type = NLA_U32 },
27 	[IFLA_BRIDGE_CFM_MEP_CONFIG_UNICAST_MAC] = NLA_POLICY_ETH_ADDR,
28 	[IFLA_BRIDGE_CFM_MEP_CONFIG_MDLEVEL]	 = NLA_POLICY_MAX(NLA_U32, 7),
29 	[IFLA_BRIDGE_CFM_MEP_CONFIG_MEPID]	 = NLA_POLICY_MAX(NLA_U32, 0x1FFF),
30 };
31 
32 static const struct nla_policy
33 br_cfm_cc_config_policy[IFLA_BRIDGE_CFM_CC_CONFIG_MAX + 1] = {
34 	[IFLA_BRIDGE_CFM_CC_CONFIG_UNSPEC]	 = { .type = NLA_REJECT },
35 	[IFLA_BRIDGE_CFM_CC_CONFIG_INSTANCE]	 = { .type = NLA_U32 },
36 	[IFLA_BRIDGE_CFM_CC_CONFIG_ENABLE]	 = { .type = NLA_U32 },
37 	[IFLA_BRIDGE_CFM_CC_CONFIG_EXP_INTERVAL] = { .type = NLA_U32 },
38 	[IFLA_BRIDGE_CFM_CC_CONFIG_EXP_MAID]	 = {
39 	.type = NLA_BINARY, .len = CFM_MAID_LENGTH },
40 };
41 
42 static const struct nla_policy
43 br_cfm_cc_peer_mep_policy[IFLA_BRIDGE_CFM_CC_PEER_MEP_MAX + 1] = {
44 	[IFLA_BRIDGE_CFM_CC_PEER_MEP_UNSPEC]	= { .type = NLA_REJECT },
45 	[IFLA_BRIDGE_CFM_CC_PEER_MEP_INSTANCE]	= { .type = NLA_U32 },
46 	[IFLA_BRIDGE_CFM_CC_PEER_MEPID]		= NLA_POLICY_MAX(NLA_U32, 0x1FFF),
47 };
48 
49 static const struct nla_policy
50 br_cfm_cc_rdi_policy[IFLA_BRIDGE_CFM_CC_RDI_MAX + 1] = {
51 	[IFLA_BRIDGE_CFM_CC_RDI_UNSPEC]		= { .type = NLA_REJECT },
52 	[IFLA_BRIDGE_CFM_CC_RDI_INSTANCE]	= { .type = NLA_U32 },
53 	[IFLA_BRIDGE_CFM_CC_RDI_RDI]		= { .type = NLA_U32 },
54 };
55 
56 static const struct nla_policy
57 br_cfm_cc_ccm_tx_policy[IFLA_BRIDGE_CFM_CC_CCM_TX_MAX + 1] = {
58 	[IFLA_BRIDGE_CFM_CC_CCM_TX_UNSPEC]	   = { .type = NLA_REJECT },
59 	[IFLA_BRIDGE_CFM_CC_CCM_TX_INSTANCE]	   = { .type = NLA_U32 },
60 	[IFLA_BRIDGE_CFM_CC_CCM_TX_DMAC]	   = NLA_POLICY_ETH_ADDR,
61 	[IFLA_BRIDGE_CFM_CC_CCM_TX_SEQ_NO_UPDATE]  = { .type = NLA_U32 },
62 	[IFLA_BRIDGE_CFM_CC_CCM_TX_PERIOD]	   = { .type = NLA_U32 },
63 	[IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV]	   = { .type = NLA_U32 },
64 	[IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV_VALUE]   = { .type = NLA_U8 },
65 	[IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV]	   = { .type = NLA_U32 },
66 	[IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV_VALUE] = { .type = NLA_U8 },
67 };
68 
69 static const struct nla_policy
70 br_cfm_policy[IFLA_BRIDGE_CFM_MAX + 1] = {
71 	[IFLA_BRIDGE_CFM_UNSPEC]		= { .type = NLA_REJECT },
72 	[IFLA_BRIDGE_CFM_MEP_CREATE]		=
73 				NLA_POLICY_NESTED(br_cfm_mep_create_policy),
74 	[IFLA_BRIDGE_CFM_MEP_DELETE]		=
75 				NLA_POLICY_NESTED(br_cfm_mep_delete_policy),
76 	[IFLA_BRIDGE_CFM_MEP_CONFIG]		=
77 				NLA_POLICY_NESTED(br_cfm_mep_config_policy),
78 	[IFLA_BRIDGE_CFM_CC_CONFIG]		=
79 				NLA_POLICY_NESTED(br_cfm_cc_config_policy),
80 	[IFLA_BRIDGE_CFM_CC_PEER_MEP_ADD]	=
81 				NLA_POLICY_NESTED(br_cfm_cc_peer_mep_policy),
82 	[IFLA_BRIDGE_CFM_CC_PEER_MEP_REMOVE]	=
83 				NLA_POLICY_NESTED(br_cfm_cc_peer_mep_policy),
84 	[IFLA_BRIDGE_CFM_CC_RDI]		=
85 				NLA_POLICY_NESTED(br_cfm_cc_rdi_policy),
86 	[IFLA_BRIDGE_CFM_CC_CCM_TX]		=
87 				NLA_POLICY_NESTED(br_cfm_cc_ccm_tx_policy),
88 };
89 
90 static int br_mep_create_parse(struct net_bridge *br, struct nlattr *attr,
91 			       struct netlink_ext_ack *extack)
92 {
93 	struct nlattr *tb[IFLA_BRIDGE_CFM_MEP_CREATE_MAX + 1];
94 	struct br_cfm_mep_create create;
95 	u32 instance;
96 	int err;
97 
98 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_MEP_CREATE_MAX, attr,
99 			       br_cfm_mep_create_policy, extack);
100 	if (err)
101 		return err;
102 
103 	if (!tb[IFLA_BRIDGE_CFM_MEP_CREATE_INSTANCE]) {
104 		NL_SET_ERR_MSG_MOD(extack, "Missing INSTANCE attribute");
105 		return -EINVAL;
106 	}
107 	if (!tb[IFLA_BRIDGE_CFM_MEP_CREATE_DOMAIN]) {
108 		NL_SET_ERR_MSG_MOD(extack, "Missing DOMAIN attribute");
109 		return -EINVAL;
110 	}
111 	if (!tb[IFLA_BRIDGE_CFM_MEP_CREATE_DIRECTION]) {
112 		NL_SET_ERR_MSG_MOD(extack, "Missing DIRECTION attribute");
113 		return -EINVAL;
114 	}
115 	if (!tb[IFLA_BRIDGE_CFM_MEP_CREATE_IFINDEX]) {
116 		NL_SET_ERR_MSG_MOD(extack, "Missing IFINDEX attribute");
117 		return -EINVAL;
118 	}
119 
120 	memset(&create, 0, sizeof(create));
121 
122 	instance =  nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_CREATE_INSTANCE]);
123 	create.domain = nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_CREATE_DOMAIN]);
124 	create.direction = nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_CREATE_DIRECTION]);
125 	create.ifindex = nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_CREATE_IFINDEX]);
126 
127 	return br_cfm_mep_create(br, instance, &create, extack);
128 }
129 
130 static int br_mep_delete_parse(struct net_bridge *br, struct nlattr *attr,
131 			       struct netlink_ext_ack *extack)
132 {
133 	struct nlattr *tb[IFLA_BRIDGE_CFM_MEP_DELETE_MAX + 1];
134 	u32 instance;
135 	int err;
136 
137 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_MEP_DELETE_MAX, attr,
138 			       br_cfm_mep_delete_policy, extack);
139 	if (err)
140 		return err;
141 
142 	if (!tb[IFLA_BRIDGE_CFM_MEP_DELETE_INSTANCE]) {
143 		NL_SET_ERR_MSG_MOD(extack,
144 				   "Missing INSTANCE attribute");
145 		return -EINVAL;
146 	}
147 
148 	instance =  nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_DELETE_INSTANCE]);
149 
150 	return br_cfm_mep_delete(br, instance, extack);
151 }
152 
153 static int br_mep_config_parse(struct net_bridge *br, struct nlattr *attr,
154 			       struct netlink_ext_ack *extack)
155 {
156 	struct nlattr *tb[IFLA_BRIDGE_CFM_MEP_CONFIG_MAX + 1];
157 	struct br_cfm_mep_config config;
158 	u32 instance;
159 	int err;
160 
161 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_MEP_CONFIG_MAX, attr,
162 			       br_cfm_mep_config_policy, extack);
163 	if (err)
164 		return err;
165 
166 	if (!tb[IFLA_BRIDGE_CFM_MEP_CONFIG_INSTANCE]) {
167 		NL_SET_ERR_MSG_MOD(extack, "Missing INSTANCE attribute");
168 		return -EINVAL;
169 	}
170 	if (!tb[IFLA_BRIDGE_CFM_MEP_CONFIG_UNICAST_MAC]) {
171 		NL_SET_ERR_MSG_MOD(extack, "Missing UNICAST_MAC attribute");
172 		return -EINVAL;
173 	}
174 	if (!tb[IFLA_BRIDGE_CFM_MEP_CONFIG_MDLEVEL]) {
175 		NL_SET_ERR_MSG_MOD(extack, "Missing MDLEVEL attribute");
176 		return -EINVAL;
177 	}
178 	if (!tb[IFLA_BRIDGE_CFM_MEP_CONFIG_MEPID]) {
179 		NL_SET_ERR_MSG_MOD(extack, "Missing MEPID attribute");
180 		return -EINVAL;
181 	}
182 
183 	memset(&config, 0, sizeof(config));
184 
185 	instance =  nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_CONFIG_INSTANCE]);
186 	nla_memcpy(&config.unicast_mac.addr,
187 		   tb[IFLA_BRIDGE_CFM_MEP_CONFIG_UNICAST_MAC],
188 		   sizeof(config.unicast_mac.addr));
189 	config.mdlevel = nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_CONFIG_MDLEVEL]);
190 	config.mepid = nla_get_u32(tb[IFLA_BRIDGE_CFM_MEP_CONFIG_MEPID]);
191 
192 	return br_cfm_mep_config_set(br, instance, &config, extack);
193 }
194 
195 static int br_cc_config_parse(struct net_bridge *br, struct nlattr *attr,
196 			      struct netlink_ext_ack *extack)
197 {
198 	struct nlattr *tb[IFLA_BRIDGE_CFM_CC_CONFIG_MAX + 1];
199 	struct br_cfm_cc_config config;
200 	u32 instance;
201 	int err;
202 
203 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_CC_CONFIG_MAX, attr,
204 			       br_cfm_cc_config_policy, extack);
205 	if (err)
206 		return err;
207 
208 	if (!tb[IFLA_BRIDGE_CFM_CC_CONFIG_INSTANCE]) {
209 		NL_SET_ERR_MSG_MOD(extack, "Missing INSTANCE attribute");
210 		return -EINVAL;
211 	}
212 	if (!tb[IFLA_BRIDGE_CFM_CC_CONFIG_ENABLE]) {
213 		NL_SET_ERR_MSG_MOD(extack, "Missing ENABLE attribute");
214 		return -EINVAL;
215 	}
216 	if (!tb[IFLA_BRIDGE_CFM_CC_CONFIG_EXP_INTERVAL]) {
217 		NL_SET_ERR_MSG_MOD(extack, "Missing INTERVAL attribute");
218 		return -EINVAL;
219 	}
220 	if (!tb[IFLA_BRIDGE_CFM_CC_CONFIG_EXP_MAID]) {
221 		NL_SET_ERR_MSG_MOD(extack, "Missing MAID attribute");
222 		return -EINVAL;
223 	}
224 
225 	memset(&config, 0, sizeof(config));
226 
227 	instance =  nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_CONFIG_INSTANCE]);
228 	config.enable = nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_CONFIG_ENABLE]);
229 	config.exp_interval = nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_CONFIG_EXP_INTERVAL]);
230 	nla_memcpy(&config.exp_maid.data, tb[IFLA_BRIDGE_CFM_CC_CONFIG_EXP_MAID],
231 		   sizeof(config.exp_maid.data));
232 
233 	return br_cfm_cc_config_set(br, instance, &config, extack);
234 }
235 
236 static int br_cc_peer_mep_add_parse(struct net_bridge *br, struct nlattr *attr,
237 				    struct netlink_ext_ack *extack)
238 {
239 	struct nlattr *tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_MAX + 1];
240 	u32 instance, peer_mep_id;
241 	int err;
242 
243 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_CC_PEER_MEP_MAX, attr,
244 			       br_cfm_cc_peer_mep_policy, extack);
245 	if (err)
246 		return err;
247 
248 	if (!tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_INSTANCE]) {
249 		NL_SET_ERR_MSG_MOD(extack, "Missing INSTANCE attribute");
250 		return -EINVAL;
251 	}
252 	if (!tb[IFLA_BRIDGE_CFM_CC_PEER_MEPID]) {
253 		NL_SET_ERR_MSG_MOD(extack, "Missing PEER_MEP_ID attribute");
254 		return -EINVAL;
255 	}
256 
257 	instance =  nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_INSTANCE]);
258 	peer_mep_id =  nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_PEER_MEPID]);
259 
260 	return br_cfm_cc_peer_mep_add(br, instance, peer_mep_id, extack);
261 }
262 
263 static int br_cc_peer_mep_remove_parse(struct net_bridge *br, struct nlattr *attr,
264 				       struct netlink_ext_ack *extack)
265 {
266 	struct nlattr *tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_MAX + 1];
267 	u32 instance, peer_mep_id;
268 	int err;
269 
270 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_CC_PEER_MEP_MAX, attr,
271 			       br_cfm_cc_peer_mep_policy, extack);
272 	if (err)
273 		return err;
274 
275 	if (!tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_INSTANCE]) {
276 		NL_SET_ERR_MSG_MOD(extack, "Missing INSTANCE attribute");
277 		return -EINVAL;
278 	}
279 	if (!tb[IFLA_BRIDGE_CFM_CC_PEER_MEPID]) {
280 		NL_SET_ERR_MSG_MOD(extack, "Missing PEER_MEP_ID attribute");
281 		return -EINVAL;
282 	}
283 
284 	instance =  nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_INSTANCE]);
285 	peer_mep_id =  nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_PEER_MEPID]);
286 
287 	return br_cfm_cc_peer_mep_remove(br, instance, peer_mep_id, extack);
288 }
289 
290 static int br_cc_rdi_parse(struct net_bridge *br, struct nlattr *attr,
291 			   struct netlink_ext_ack *extack)
292 {
293 	struct nlattr *tb[IFLA_BRIDGE_CFM_CC_RDI_MAX + 1];
294 	u32 instance, rdi;
295 	int err;
296 
297 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_CC_RDI_MAX, attr,
298 			       br_cfm_cc_rdi_policy, extack);
299 	if (err)
300 		return err;
301 
302 	if (!tb[IFLA_BRIDGE_CFM_CC_RDI_INSTANCE]) {
303 		NL_SET_ERR_MSG_MOD(extack, "Missing INSTANCE attribute");
304 		return -EINVAL;
305 	}
306 	if (!tb[IFLA_BRIDGE_CFM_CC_RDI_RDI]) {
307 		NL_SET_ERR_MSG_MOD(extack, "Missing RDI attribute");
308 		return -EINVAL;
309 	}
310 
311 	instance =  nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_RDI_INSTANCE]);
312 	rdi =  nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_RDI_RDI]);
313 
314 	return br_cfm_cc_rdi_set(br, instance, rdi, extack);
315 }
316 
317 static int br_cc_ccm_tx_parse(struct net_bridge *br, struct nlattr *attr,
318 			      struct netlink_ext_ack *extack)
319 {
320 	struct nlattr *tb[IFLA_BRIDGE_CFM_CC_CCM_TX_MAX + 1];
321 	struct br_cfm_cc_ccm_tx_info tx_info;
322 	u32 instance;
323 	int err;
324 
325 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_CC_CCM_TX_MAX, attr,
326 			       br_cfm_cc_ccm_tx_policy, extack);
327 	if (err)
328 		return err;
329 
330 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_INSTANCE]) {
331 		NL_SET_ERR_MSG_MOD(extack, "Missing INSTANCE attribute");
332 		return -EINVAL;
333 	}
334 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_DMAC]) {
335 		NL_SET_ERR_MSG_MOD(extack, "Missing DMAC attribute");
336 		return -EINVAL;
337 	}
338 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_SEQ_NO_UPDATE]) {
339 		NL_SET_ERR_MSG_MOD(extack, "Missing SEQ_NO_UPDATE attribute");
340 		return -EINVAL;
341 	}
342 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_PERIOD]) {
343 		NL_SET_ERR_MSG_MOD(extack, "Missing PERIOD attribute");
344 		return -EINVAL;
345 	}
346 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV]) {
347 		NL_SET_ERR_MSG_MOD(extack, "Missing IF_TLV attribute");
348 		return -EINVAL;
349 	}
350 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV_VALUE]) {
351 		NL_SET_ERR_MSG_MOD(extack, "Missing IF_TLV_VALUE attribute");
352 		return -EINVAL;
353 	}
354 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV]) {
355 		NL_SET_ERR_MSG_MOD(extack, "Missing PORT_TLV attribute");
356 		return -EINVAL;
357 	}
358 	if (!tb[IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV_VALUE]) {
359 		NL_SET_ERR_MSG_MOD(extack, "Missing PORT_TLV_VALUE attribute");
360 		return -EINVAL;
361 	}
362 
363 	memset(&tx_info, 0, sizeof(tx_info));
364 
365 	instance = nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_RDI_INSTANCE]);
366 	nla_memcpy(&tx_info.dmac.addr,
367 		   tb[IFLA_BRIDGE_CFM_CC_CCM_TX_DMAC],
368 		   sizeof(tx_info.dmac.addr));
369 	tx_info.seq_no_update = nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_CCM_TX_SEQ_NO_UPDATE]);
370 	tx_info.period = nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_CCM_TX_PERIOD]);
371 	tx_info.if_tlv = nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV]);
372 	tx_info.if_tlv_value = nla_get_u8(tb[IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV_VALUE]);
373 	tx_info.port_tlv = nla_get_u32(tb[IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV]);
374 	tx_info.port_tlv_value = nla_get_u8(tb[IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV_VALUE]);
375 
376 	return br_cfm_cc_ccm_tx(br, instance, &tx_info, extack);
377 }
378 
379 int br_cfm_parse(struct net_bridge *br, struct net_bridge_port *p,
380 		 struct nlattr *attr, int cmd, struct netlink_ext_ack *extack)
381 {
382 	struct nlattr *tb[IFLA_BRIDGE_CFM_MAX + 1];
383 	int err;
384 
385 	/* When this function is called for a port then the br pointer is
386 	 * invalid, therefor set the br to point correctly
387 	 */
388 	if (p)
389 		br = p->br;
390 
391 	err = nla_parse_nested(tb, IFLA_BRIDGE_CFM_MAX, attr,
392 			       br_cfm_policy, extack);
393 	if (err)
394 		return err;
395 
396 	if (tb[IFLA_BRIDGE_CFM_MEP_CREATE]) {
397 		err = br_mep_create_parse(br, tb[IFLA_BRIDGE_CFM_MEP_CREATE],
398 					  extack);
399 		if (err)
400 			return err;
401 	}
402 
403 	if (tb[IFLA_BRIDGE_CFM_MEP_DELETE]) {
404 		err = br_mep_delete_parse(br, tb[IFLA_BRIDGE_CFM_MEP_DELETE],
405 					  extack);
406 		if (err)
407 			return err;
408 	}
409 
410 	if (tb[IFLA_BRIDGE_CFM_MEP_CONFIG]) {
411 		err = br_mep_config_parse(br, tb[IFLA_BRIDGE_CFM_MEP_CONFIG],
412 					  extack);
413 		if (err)
414 			return err;
415 	}
416 
417 	if (tb[IFLA_BRIDGE_CFM_CC_CONFIG]) {
418 		err = br_cc_config_parse(br, tb[IFLA_BRIDGE_CFM_CC_CONFIG],
419 					 extack);
420 		if (err)
421 			return err;
422 	}
423 
424 	if (tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_ADD]) {
425 		err = br_cc_peer_mep_add_parse(br, tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_ADD],
426 					       extack);
427 		if (err)
428 			return err;
429 	}
430 
431 	if (tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_REMOVE]) {
432 		err = br_cc_peer_mep_remove_parse(br, tb[IFLA_BRIDGE_CFM_CC_PEER_MEP_REMOVE],
433 						  extack);
434 		if (err)
435 			return err;
436 	}
437 
438 	if (tb[IFLA_BRIDGE_CFM_CC_RDI]) {
439 		err = br_cc_rdi_parse(br, tb[IFLA_BRIDGE_CFM_CC_RDI],
440 				      extack);
441 		if (err)
442 			return err;
443 	}
444 
445 	if (tb[IFLA_BRIDGE_CFM_CC_CCM_TX]) {
446 		err = br_cc_ccm_tx_parse(br, tb[IFLA_BRIDGE_CFM_CC_CCM_TX],
447 					 extack);
448 		if (err)
449 			return err;
450 	}
451 
452 	return 0;
453 }
454 
455 int br_cfm_config_fill_info(struct sk_buff *skb, struct net_bridge *br)
456 {
457 	struct br_cfm_peer_mep *peer_mep;
458 	struct br_cfm_mep *mep;
459 	struct nlattr *tb;
460 
461 	hlist_for_each_entry_rcu(mep, &br->mep_list, head) {
462 		tb = nla_nest_start(skb, IFLA_BRIDGE_CFM_MEP_CREATE_INFO);
463 		if (!tb)
464 			goto nla_info_failure;
465 
466 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_CREATE_INSTANCE,
467 				mep->instance))
468 			goto nla_put_failure;
469 
470 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_CREATE_DOMAIN,
471 				mep->create.domain))
472 			goto nla_put_failure;
473 
474 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_CREATE_DIRECTION,
475 				mep->create.direction))
476 			goto nla_put_failure;
477 
478 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_CREATE_IFINDEX,
479 				mep->create.ifindex))
480 			goto nla_put_failure;
481 
482 		nla_nest_end(skb, tb);
483 
484 		tb = nla_nest_start(skb, IFLA_BRIDGE_CFM_MEP_CONFIG_INFO);
485 
486 		if (!tb)
487 			goto nla_info_failure;
488 
489 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_CONFIG_INSTANCE,
490 				mep->instance))
491 			goto nla_put_failure;
492 
493 		if (nla_put(skb, IFLA_BRIDGE_CFM_MEP_CONFIG_UNICAST_MAC,
494 			    sizeof(mep->config.unicast_mac.addr),
495 			    mep->config.unicast_mac.addr))
496 			goto nla_put_failure;
497 
498 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_CONFIG_MDLEVEL,
499 				mep->config.mdlevel))
500 			goto nla_put_failure;
501 
502 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_CONFIG_MEPID,
503 				mep->config.mepid))
504 			goto nla_put_failure;
505 
506 		nla_nest_end(skb, tb);
507 
508 		tb = nla_nest_start(skb, IFLA_BRIDGE_CFM_CC_CONFIG_INFO);
509 
510 		if (!tb)
511 			goto nla_info_failure;
512 
513 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CONFIG_INSTANCE,
514 				mep->instance))
515 			goto nla_put_failure;
516 
517 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CONFIG_ENABLE,
518 				mep->cc_config.enable))
519 			goto nla_put_failure;
520 
521 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CONFIG_EXP_INTERVAL,
522 				mep->cc_config.exp_interval))
523 			goto nla_put_failure;
524 
525 		if (nla_put(skb, IFLA_BRIDGE_CFM_CC_CONFIG_EXP_MAID,
526 			    sizeof(mep->cc_config.exp_maid.data),
527 			    mep->cc_config.exp_maid.data))
528 			goto nla_put_failure;
529 
530 		nla_nest_end(skb, tb);
531 
532 		tb = nla_nest_start(skb, IFLA_BRIDGE_CFM_CC_RDI_INFO);
533 
534 		if (!tb)
535 			goto nla_info_failure;
536 
537 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_RDI_INSTANCE,
538 				mep->instance))
539 			goto nla_put_failure;
540 
541 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_RDI_RDI,
542 				mep->rdi))
543 			goto nla_put_failure;
544 
545 		nla_nest_end(skb, tb);
546 
547 		tb = nla_nest_start(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_INFO);
548 
549 		if (!tb)
550 			goto nla_info_failure;
551 
552 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_INSTANCE,
553 				mep->instance))
554 			goto nla_put_failure;
555 
556 		if (nla_put(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_DMAC,
557 			    sizeof(mep->cc_ccm_tx_info.dmac),
558 			    mep->cc_ccm_tx_info.dmac.addr))
559 			goto nla_put_failure;
560 
561 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_SEQ_NO_UPDATE,
562 				mep->cc_ccm_tx_info.seq_no_update))
563 			goto nla_put_failure;
564 
565 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_PERIOD,
566 				mep->cc_ccm_tx_info.period))
567 			goto nla_put_failure;
568 
569 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV,
570 				mep->cc_ccm_tx_info.if_tlv))
571 			goto nla_put_failure;
572 
573 		if (nla_put_u8(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_IF_TLV_VALUE,
574 			       mep->cc_ccm_tx_info.if_tlv_value))
575 			goto nla_put_failure;
576 
577 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV,
578 				mep->cc_ccm_tx_info.port_tlv))
579 			goto nla_put_failure;
580 
581 		if (nla_put_u8(skb, IFLA_BRIDGE_CFM_CC_CCM_TX_PORT_TLV_VALUE,
582 			       mep->cc_ccm_tx_info.port_tlv_value))
583 			goto nla_put_failure;
584 
585 		nla_nest_end(skb, tb);
586 
587 		hlist_for_each_entry_rcu(peer_mep, &mep->peer_mep_list, head) {
588 			tb = nla_nest_start(skb,
589 					    IFLA_BRIDGE_CFM_CC_PEER_MEP_INFO);
590 
591 			if (!tb)
592 				goto nla_info_failure;
593 
594 			if (nla_put_u32(skb,
595 					IFLA_BRIDGE_CFM_CC_PEER_MEP_INSTANCE,
596 					mep->instance))
597 				goto nla_put_failure;
598 
599 			if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_PEER_MEPID,
600 					peer_mep->mepid))
601 				goto nla_put_failure;
602 
603 			nla_nest_end(skb, tb);
604 		}
605 	}
606 
607 	return 0;
608 
609 nla_put_failure:
610 	nla_nest_cancel(skb, tb);
611 
612 nla_info_failure:
613 	return -EMSGSIZE;
614 }
615 
616 int br_cfm_status_fill_info(struct sk_buff *skb,
617 			    struct net_bridge *br,
618 			    bool getlink)
619 {
620 	struct br_cfm_peer_mep *peer_mep;
621 	struct br_cfm_mep *mep;
622 	struct nlattr *tb;
623 
624 	hlist_for_each_entry_rcu(mep, &br->mep_list, head) {
625 		tb = nla_nest_start(skb, IFLA_BRIDGE_CFM_MEP_STATUS_INFO);
626 		if (!tb)
627 			goto nla_info_failure;
628 
629 		if (nla_put_u32(skb, IFLA_BRIDGE_CFM_MEP_STATUS_INSTANCE,
630 				mep->instance))
631 			goto nla_put_failure;
632 
633 		if (nla_put_u32(skb,
634 				IFLA_BRIDGE_CFM_MEP_STATUS_OPCODE_UNEXP_SEEN,
635 				mep->status.opcode_unexp_seen))
636 			goto nla_put_failure;
637 
638 		if (nla_put_u32(skb,
639 				IFLA_BRIDGE_CFM_MEP_STATUS_VERSION_UNEXP_SEEN,
640 				mep->status.version_unexp_seen))
641 			goto nla_put_failure;
642 
643 		if (nla_put_u32(skb,
644 				IFLA_BRIDGE_CFM_MEP_STATUS_RX_LEVEL_LOW_SEEN,
645 				mep->status.rx_level_low_seen))
646 			goto nla_put_failure;
647 
648 		/* Only clear if this is a GETLINK */
649 		if (getlink) {
650 			/* Clear all 'seen' indications */
651 			mep->status.opcode_unexp_seen = false;
652 			mep->status.version_unexp_seen = false;
653 			mep->status.rx_level_low_seen = false;
654 		}
655 
656 		nla_nest_end(skb, tb);
657 
658 		hlist_for_each_entry_rcu(peer_mep, &mep->peer_mep_list, head) {
659 			tb = nla_nest_start(skb,
660 					    IFLA_BRIDGE_CFM_CC_PEER_STATUS_INFO);
661 			if (!tb)
662 				goto nla_info_failure;
663 
664 			if (nla_put_u32(skb,
665 					IFLA_BRIDGE_CFM_CC_PEER_STATUS_INSTANCE,
666 					mep->instance))
667 				goto nla_put_failure;
668 
669 			if (nla_put_u32(skb,
670 					IFLA_BRIDGE_CFM_CC_PEER_STATUS_PEER_MEPID,
671 					peer_mep->mepid))
672 				goto nla_put_failure;
673 
674 			if (nla_put_u32(skb,
675 					IFLA_BRIDGE_CFM_CC_PEER_STATUS_CCM_DEFECT,
676 					peer_mep->cc_status.ccm_defect))
677 				goto nla_put_failure;
678 
679 			if (nla_put_u32(skb, IFLA_BRIDGE_CFM_CC_PEER_STATUS_RDI,
680 					peer_mep->cc_status.rdi))
681 				goto nla_put_failure;
682 
683 			if (nla_put_u8(skb,
684 				       IFLA_BRIDGE_CFM_CC_PEER_STATUS_PORT_TLV_VALUE,
685 				       peer_mep->cc_status.port_tlv_value))
686 				goto nla_put_failure;
687 
688 			if (nla_put_u8(skb,
689 				       IFLA_BRIDGE_CFM_CC_PEER_STATUS_IF_TLV_VALUE,
690 				       peer_mep->cc_status.if_tlv_value))
691 				goto nla_put_failure;
692 
693 			if (nla_put_u32(skb,
694 					IFLA_BRIDGE_CFM_CC_PEER_STATUS_SEEN,
695 					peer_mep->cc_status.seen))
696 				goto nla_put_failure;
697 
698 			if (nla_put_u32(skb,
699 					IFLA_BRIDGE_CFM_CC_PEER_STATUS_TLV_SEEN,
700 					peer_mep->cc_status.tlv_seen))
701 				goto nla_put_failure;
702 
703 			if (nla_put_u32(skb,
704 					IFLA_BRIDGE_CFM_CC_PEER_STATUS_SEQ_UNEXP_SEEN,
705 					peer_mep->cc_status.seq_unexp_seen))
706 				goto nla_put_failure;
707 
708 			if (getlink) { /* Only clear if this is a GETLINK */
709 				/* Clear all 'seen' indications */
710 				peer_mep->cc_status.seen = false;
711 				peer_mep->cc_status.tlv_seen = false;
712 				peer_mep->cc_status.seq_unexp_seen = false;
713 			}
714 
715 			nla_nest_end(skb, tb);
716 		}
717 	}
718 
719 	return 0;
720 
721 nla_put_failure:
722 	nla_nest_cancel(skb, tb);
723 
724 nla_info_failure:
725 	return -EMSGSIZE;
726 }
727