xref: /linux/net/bridge/br_mrp_netlink.c (revision eb01fe7abbe2d0b38824d2a93fdb4cc3eaf2ccc1)
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 
3 #include <net/genetlink.h>
4 
5 #include <uapi/linux/mrp_bridge.h>
6 #include "br_private.h"
7 #include "br_private_mrp.h"
8 
9 static const struct nla_policy br_mrp_policy[IFLA_BRIDGE_MRP_MAX + 1] = {
10 	[IFLA_BRIDGE_MRP_UNSPEC]	= { .type = NLA_REJECT },
11 	[IFLA_BRIDGE_MRP_INSTANCE]	= { .type = NLA_NESTED },
12 	[IFLA_BRIDGE_MRP_PORT_STATE]	= { .type = NLA_NESTED },
13 	[IFLA_BRIDGE_MRP_PORT_ROLE]	= { .type = NLA_NESTED },
14 	[IFLA_BRIDGE_MRP_RING_STATE]	= { .type = NLA_NESTED },
15 	[IFLA_BRIDGE_MRP_RING_ROLE]	= { .type = NLA_NESTED },
16 	[IFLA_BRIDGE_MRP_START_TEST]	= { .type = NLA_NESTED },
17 	[IFLA_BRIDGE_MRP_IN_ROLE]	= { .type = NLA_NESTED },
18 	[IFLA_BRIDGE_MRP_IN_STATE]	= { .type = NLA_NESTED },
19 	[IFLA_BRIDGE_MRP_START_IN_TEST]	= { .type = NLA_NESTED },
20 };
21 
22 static const struct nla_policy
23 br_mrp_instance_policy[IFLA_BRIDGE_MRP_INSTANCE_MAX + 1] = {
24 	[IFLA_BRIDGE_MRP_INSTANCE_UNSPEC]	= { .type = NLA_REJECT },
25 	[IFLA_BRIDGE_MRP_INSTANCE_RING_ID]	= { .type = NLA_U32 },
26 	[IFLA_BRIDGE_MRP_INSTANCE_P_IFINDEX]	= { .type = NLA_U32 },
27 	[IFLA_BRIDGE_MRP_INSTANCE_S_IFINDEX]	= { .type = NLA_U32 },
28 	[IFLA_BRIDGE_MRP_INSTANCE_PRIO]		= { .type = NLA_U16 },
29 };
30 
31 static int br_mrp_instance_parse(struct net_bridge *br, struct nlattr *attr,
32 				 int cmd, struct netlink_ext_ack *extack)
33 {
34 	struct nlattr *tb[IFLA_BRIDGE_MRP_INSTANCE_MAX + 1];
35 	struct br_mrp_instance inst;
36 	int err;
37 
38 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_INSTANCE_MAX, attr,
39 			       br_mrp_instance_policy, extack);
40 	if (err)
41 		return err;
42 
43 	if (!tb[IFLA_BRIDGE_MRP_INSTANCE_RING_ID] ||
44 	    !tb[IFLA_BRIDGE_MRP_INSTANCE_P_IFINDEX] ||
45 	    !tb[IFLA_BRIDGE_MRP_INSTANCE_S_IFINDEX]) {
46 		NL_SET_ERR_MSG_MOD(extack,
47 				   "Missing attribute: RING_ID or P_IFINDEX or S_IFINDEX");
48 		return -EINVAL;
49 	}
50 
51 	memset(&inst, 0, sizeof(inst));
52 
53 	inst.ring_id = nla_get_u32(tb[IFLA_BRIDGE_MRP_INSTANCE_RING_ID]);
54 	inst.p_ifindex = nla_get_u32(tb[IFLA_BRIDGE_MRP_INSTANCE_P_IFINDEX]);
55 	inst.s_ifindex = nla_get_u32(tb[IFLA_BRIDGE_MRP_INSTANCE_S_IFINDEX]);
56 	inst.prio = MRP_DEFAULT_PRIO;
57 
58 	if (tb[IFLA_BRIDGE_MRP_INSTANCE_PRIO])
59 		inst.prio = nla_get_u16(tb[IFLA_BRIDGE_MRP_INSTANCE_PRIO]);
60 
61 	if (cmd == RTM_SETLINK)
62 		return br_mrp_add(br, &inst);
63 	else
64 		return br_mrp_del(br, &inst);
65 
66 	return 0;
67 }
68 
69 static const struct nla_policy
70 br_mrp_port_state_policy[IFLA_BRIDGE_MRP_PORT_STATE_MAX + 1] = {
71 	[IFLA_BRIDGE_MRP_PORT_STATE_UNSPEC]	= { .type = NLA_REJECT },
72 	[IFLA_BRIDGE_MRP_PORT_STATE_STATE]	= { .type = NLA_U32 },
73 };
74 
75 static int br_mrp_port_state_parse(struct net_bridge_port *p,
76 				   struct nlattr *attr,
77 				   struct netlink_ext_ack *extack)
78 {
79 	struct nlattr *tb[IFLA_BRIDGE_MRP_PORT_STATE_MAX + 1];
80 	enum br_mrp_port_state_type state;
81 	int err;
82 
83 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_PORT_STATE_MAX, attr,
84 			       br_mrp_port_state_policy, extack);
85 	if (err)
86 		return err;
87 
88 	if (!tb[IFLA_BRIDGE_MRP_PORT_STATE_STATE]) {
89 		NL_SET_ERR_MSG_MOD(extack, "Missing attribute: STATE");
90 		return -EINVAL;
91 	}
92 
93 	state = nla_get_u32(tb[IFLA_BRIDGE_MRP_PORT_STATE_STATE]);
94 
95 	return br_mrp_set_port_state(p, state);
96 }
97 
98 static const struct nla_policy
99 br_mrp_port_role_policy[IFLA_BRIDGE_MRP_PORT_ROLE_MAX + 1] = {
100 	[IFLA_BRIDGE_MRP_PORT_ROLE_UNSPEC]	= { .type = NLA_REJECT },
101 	[IFLA_BRIDGE_MRP_PORT_ROLE_ROLE]	= { .type = NLA_U32 },
102 };
103 
104 static int br_mrp_port_role_parse(struct net_bridge_port *p,
105 				  struct nlattr *attr,
106 				  struct netlink_ext_ack *extack)
107 {
108 	struct nlattr *tb[IFLA_BRIDGE_MRP_PORT_ROLE_MAX + 1];
109 	enum br_mrp_port_role_type role;
110 	int err;
111 
112 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_PORT_ROLE_MAX, attr,
113 			       br_mrp_port_role_policy, extack);
114 	if (err)
115 		return err;
116 
117 	if (!tb[IFLA_BRIDGE_MRP_PORT_ROLE_ROLE]) {
118 		NL_SET_ERR_MSG_MOD(extack, "Missing attribute: ROLE");
119 		return -EINVAL;
120 	}
121 
122 	role = nla_get_u32(tb[IFLA_BRIDGE_MRP_PORT_ROLE_ROLE]);
123 
124 	return br_mrp_set_port_role(p, role);
125 }
126 
127 static const struct nla_policy
128 br_mrp_ring_state_policy[IFLA_BRIDGE_MRP_RING_STATE_MAX + 1] = {
129 	[IFLA_BRIDGE_MRP_RING_STATE_UNSPEC]	= { .type = NLA_REJECT },
130 	[IFLA_BRIDGE_MRP_RING_STATE_RING_ID]	= { .type = NLA_U32 },
131 	[IFLA_BRIDGE_MRP_RING_STATE_STATE]	= { .type = NLA_U32 },
132 };
133 
134 static int br_mrp_ring_state_parse(struct net_bridge *br, struct nlattr *attr,
135 				   struct netlink_ext_ack *extack)
136 {
137 	struct nlattr *tb[IFLA_BRIDGE_MRP_RING_STATE_MAX + 1];
138 	struct br_mrp_ring_state state;
139 	int err;
140 
141 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_RING_STATE_MAX, attr,
142 			       br_mrp_ring_state_policy, extack);
143 	if (err)
144 		return err;
145 
146 	if (!tb[IFLA_BRIDGE_MRP_RING_STATE_RING_ID] ||
147 	    !tb[IFLA_BRIDGE_MRP_RING_STATE_STATE]) {
148 		NL_SET_ERR_MSG_MOD(extack,
149 				   "Missing attribute: RING_ID or STATE");
150 		return -EINVAL;
151 	}
152 
153 	memset(&state, 0x0, sizeof(state));
154 
155 	state.ring_id = nla_get_u32(tb[IFLA_BRIDGE_MRP_RING_STATE_RING_ID]);
156 	state.ring_state = nla_get_u32(tb[IFLA_BRIDGE_MRP_RING_STATE_STATE]);
157 
158 	return br_mrp_set_ring_state(br, &state);
159 }
160 
161 static const struct nla_policy
162 br_mrp_ring_role_policy[IFLA_BRIDGE_MRP_RING_ROLE_MAX + 1] = {
163 	[IFLA_BRIDGE_MRP_RING_ROLE_UNSPEC]	= { .type = NLA_REJECT },
164 	[IFLA_BRIDGE_MRP_RING_ROLE_RING_ID]	= { .type = NLA_U32 },
165 	[IFLA_BRIDGE_MRP_RING_ROLE_ROLE]	= { .type = NLA_U32 },
166 };
167 
168 static int br_mrp_ring_role_parse(struct net_bridge *br, struct nlattr *attr,
169 				  struct netlink_ext_ack *extack)
170 {
171 	struct nlattr *tb[IFLA_BRIDGE_MRP_RING_ROLE_MAX + 1];
172 	struct br_mrp_ring_role role;
173 	int err;
174 
175 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_RING_ROLE_MAX, attr,
176 			       br_mrp_ring_role_policy, extack);
177 	if (err)
178 		return err;
179 
180 	if (!tb[IFLA_BRIDGE_MRP_RING_ROLE_RING_ID] ||
181 	    !tb[IFLA_BRIDGE_MRP_RING_ROLE_ROLE]) {
182 		NL_SET_ERR_MSG_MOD(extack,
183 				   "Missing attribute: RING_ID or ROLE");
184 		return -EINVAL;
185 	}
186 
187 	memset(&role, 0x0, sizeof(role));
188 
189 	role.ring_id = nla_get_u32(tb[IFLA_BRIDGE_MRP_RING_ROLE_RING_ID]);
190 	role.ring_role = nla_get_u32(tb[IFLA_BRIDGE_MRP_RING_ROLE_ROLE]);
191 
192 	return br_mrp_set_ring_role(br, &role);
193 }
194 
195 static const struct nla_policy
196 br_mrp_start_test_policy[IFLA_BRIDGE_MRP_START_TEST_MAX + 1] = {
197 	[IFLA_BRIDGE_MRP_START_TEST_UNSPEC]	= { .type = NLA_REJECT },
198 	[IFLA_BRIDGE_MRP_START_TEST_RING_ID]	= { .type = NLA_U32 },
199 	[IFLA_BRIDGE_MRP_START_TEST_INTERVAL]	= { .type = NLA_U32 },
200 	[IFLA_BRIDGE_MRP_START_TEST_MAX_MISS]	= { .type = NLA_U32 },
201 	[IFLA_BRIDGE_MRP_START_TEST_PERIOD]	= { .type = NLA_U32 },
202 	[IFLA_BRIDGE_MRP_START_TEST_MONITOR]	= { .type = NLA_U32 },
203 };
204 
205 static int br_mrp_start_test_parse(struct net_bridge *br, struct nlattr *attr,
206 				   struct netlink_ext_ack *extack)
207 {
208 	struct nlattr *tb[IFLA_BRIDGE_MRP_START_TEST_MAX + 1];
209 	struct br_mrp_start_test test;
210 	int err;
211 
212 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_START_TEST_MAX, attr,
213 			       br_mrp_start_test_policy, extack);
214 	if (err)
215 		return err;
216 
217 	if (!tb[IFLA_BRIDGE_MRP_START_TEST_RING_ID] ||
218 	    !tb[IFLA_BRIDGE_MRP_START_TEST_INTERVAL] ||
219 	    !tb[IFLA_BRIDGE_MRP_START_TEST_MAX_MISS] ||
220 	    !tb[IFLA_BRIDGE_MRP_START_TEST_PERIOD]) {
221 		NL_SET_ERR_MSG_MOD(extack,
222 				   "Missing attribute: RING_ID or INTERVAL or MAX_MISS or PERIOD");
223 		return -EINVAL;
224 	}
225 
226 	memset(&test, 0x0, sizeof(test));
227 
228 	test.ring_id = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_TEST_RING_ID]);
229 	test.interval = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_TEST_INTERVAL]);
230 	test.max_miss = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_TEST_MAX_MISS]);
231 	test.period = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_TEST_PERIOD]);
232 	test.monitor = false;
233 
234 	if (tb[IFLA_BRIDGE_MRP_START_TEST_MONITOR])
235 		test.monitor =
236 			nla_get_u32(tb[IFLA_BRIDGE_MRP_START_TEST_MONITOR]);
237 
238 	return br_mrp_start_test(br, &test);
239 }
240 
241 static const struct nla_policy
242 br_mrp_in_state_policy[IFLA_BRIDGE_MRP_IN_STATE_MAX + 1] = {
243 	[IFLA_BRIDGE_MRP_IN_STATE_UNSPEC]	= { .type = NLA_REJECT },
244 	[IFLA_BRIDGE_MRP_IN_STATE_IN_ID]	= { .type = NLA_U32 },
245 	[IFLA_BRIDGE_MRP_IN_STATE_STATE]	= { .type = NLA_U32 },
246 };
247 
248 static int br_mrp_in_state_parse(struct net_bridge *br, struct nlattr *attr,
249 				 struct netlink_ext_ack *extack)
250 {
251 	struct nlattr *tb[IFLA_BRIDGE_MRP_IN_STATE_MAX + 1];
252 	struct br_mrp_in_state state;
253 	int err;
254 
255 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_IN_STATE_MAX, attr,
256 			       br_mrp_in_state_policy, extack);
257 	if (err)
258 		return err;
259 
260 	if (!tb[IFLA_BRIDGE_MRP_IN_STATE_IN_ID] ||
261 	    !tb[IFLA_BRIDGE_MRP_IN_STATE_STATE]) {
262 		NL_SET_ERR_MSG_MOD(extack,
263 				   "Missing attribute: IN_ID or STATE");
264 		return -EINVAL;
265 	}
266 
267 	memset(&state, 0x0, sizeof(state));
268 
269 	state.in_id = nla_get_u32(tb[IFLA_BRIDGE_MRP_IN_STATE_IN_ID]);
270 	state.in_state = nla_get_u32(tb[IFLA_BRIDGE_MRP_IN_STATE_STATE]);
271 
272 	return br_mrp_set_in_state(br, &state);
273 }
274 
275 static const struct nla_policy
276 br_mrp_in_role_policy[IFLA_BRIDGE_MRP_IN_ROLE_MAX + 1] = {
277 	[IFLA_BRIDGE_MRP_IN_ROLE_UNSPEC]	= { .type = NLA_REJECT },
278 	[IFLA_BRIDGE_MRP_IN_ROLE_RING_ID]	= { .type = NLA_U32 },
279 	[IFLA_BRIDGE_MRP_IN_ROLE_IN_ID]		= { .type = NLA_U16 },
280 	[IFLA_BRIDGE_MRP_IN_ROLE_ROLE]		= { .type = NLA_U32 },
281 	[IFLA_BRIDGE_MRP_IN_ROLE_I_IFINDEX]	= { .type = NLA_U32 },
282 };
283 
284 static int br_mrp_in_role_parse(struct net_bridge *br, struct nlattr *attr,
285 				struct netlink_ext_ack *extack)
286 {
287 	struct nlattr *tb[IFLA_BRIDGE_MRP_IN_ROLE_MAX + 1];
288 	struct br_mrp_in_role role;
289 	int err;
290 
291 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_IN_ROLE_MAX, attr,
292 			       br_mrp_in_role_policy, extack);
293 	if (err)
294 		return err;
295 
296 	if (!tb[IFLA_BRIDGE_MRP_IN_ROLE_RING_ID] ||
297 	    !tb[IFLA_BRIDGE_MRP_IN_ROLE_IN_ID] ||
298 	    !tb[IFLA_BRIDGE_MRP_IN_ROLE_I_IFINDEX] ||
299 	    !tb[IFLA_BRIDGE_MRP_IN_ROLE_ROLE]) {
300 		NL_SET_ERR_MSG_MOD(extack,
301 				   "Missing attribute: RING_ID or ROLE or IN_ID or I_IFINDEX");
302 		return -EINVAL;
303 	}
304 
305 	memset(&role, 0x0, sizeof(role));
306 
307 	role.ring_id = nla_get_u32(tb[IFLA_BRIDGE_MRP_IN_ROLE_RING_ID]);
308 	role.in_id = nla_get_u16(tb[IFLA_BRIDGE_MRP_IN_ROLE_IN_ID]);
309 	role.i_ifindex = nla_get_u32(tb[IFLA_BRIDGE_MRP_IN_ROLE_I_IFINDEX]);
310 	role.in_role = nla_get_u32(tb[IFLA_BRIDGE_MRP_IN_ROLE_ROLE]);
311 
312 	return br_mrp_set_in_role(br, &role);
313 }
314 
315 static const struct nla_policy
316 br_mrp_start_in_test_policy[IFLA_BRIDGE_MRP_START_IN_TEST_MAX + 1] = {
317 	[IFLA_BRIDGE_MRP_START_IN_TEST_UNSPEC]	= { .type = NLA_REJECT },
318 	[IFLA_BRIDGE_MRP_START_IN_TEST_IN_ID]	= { .type = NLA_U32 },
319 	[IFLA_BRIDGE_MRP_START_IN_TEST_INTERVAL]	= { .type = NLA_U32 },
320 	[IFLA_BRIDGE_MRP_START_IN_TEST_MAX_MISS]	= { .type = NLA_U32 },
321 	[IFLA_BRIDGE_MRP_START_IN_TEST_PERIOD]	= { .type = NLA_U32 },
322 };
323 
324 static int br_mrp_start_in_test_parse(struct net_bridge *br,
325 				      struct nlattr *attr,
326 				      struct netlink_ext_ack *extack)
327 {
328 	struct nlattr *tb[IFLA_BRIDGE_MRP_START_IN_TEST_MAX + 1];
329 	struct br_mrp_start_in_test test;
330 	int err;
331 
332 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_START_IN_TEST_MAX, attr,
333 			       br_mrp_start_in_test_policy, extack);
334 	if (err)
335 		return err;
336 
337 	if (!tb[IFLA_BRIDGE_MRP_START_IN_TEST_IN_ID] ||
338 	    !tb[IFLA_BRIDGE_MRP_START_IN_TEST_INTERVAL] ||
339 	    !tb[IFLA_BRIDGE_MRP_START_IN_TEST_MAX_MISS] ||
340 	    !tb[IFLA_BRIDGE_MRP_START_IN_TEST_PERIOD]) {
341 		NL_SET_ERR_MSG_MOD(extack,
342 				   "Missing attribute: RING_ID or INTERVAL or MAX_MISS or PERIOD");
343 		return -EINVAL;
344 	}
345 
346 	memset(&test, 0x0, sizeof(test));
347 
348 	test.in_id = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_IN_TEST_IN_ID]);
349 	test.interval = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_IN_TEST_INTERVAL]);
350 	test.max_miss = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_IN_TEST_MAX_MISS]);
351 	test.period = nla_get_u32(tb[IFLA_BRIDGE_MRP_START_IN_TEST_PERIOD]);
352 
353 	return br_mrp_start_in_test(br, &test);
354 }
355 
356 int br_mrp_parse(struct net_bridge *br, struct net_bridge_port *p,
357 		 struct nlattr *attr, int cmd, struct netlink_ext_ack *extack)
358 {
359 	struct nlattr *tb[IFLA_BRIDGE_MRP_MAX + 1];
360 	int err;
361 
362 	/* When this function is called for a port then the br pointer is
363 	 * invalid, therefor set the br to point correctly
364 	 */
365 	if (p)
366 		br = p->br;
367 
368 	if (br->stp_enabled != BR_NO_STP) {
369 		NL_SET_ERR_MSG_MOD(extack, "MRP can't be enabled if STP is already enabled");
370 		return -EINVAL;
371 	}
372 
373 	err = nla_parse_nested(tb, IFLA_BRIDGE_MRP_MAX, attr,
374 			       br_mrp_policy, extack);
375 	if (err)
376 		return err;
377 
378 	if (tb[IFLA_BRIDGE_MRP_INSTANCE]) {
379 		err = br_mrp_instance_parse(br, tb[IFLA_BRIDGE_MRP_INSTANCE],
380 					    cmd, extack);
381 		if (err)
382 			return err;
383 	}
384 
385 	if (tb[IFLA_BRIDGE_MRP_PORT_STATE]) {
386 		err = br_mrp_port_state_parse(p, tb[IFLA_BRIDGE_MRP_PORT_STATE],
387 					      extack);
388 		if (err)
389 			return err;
390 	}
391 
392 	if (tb[IFLA_BRIDGE_MRP_PORT_ROLE]) {
393 		err = br_mrp_port_role_parse(p, tb[IFLA_BRIDGE_MRP_PORT_ROLE],
394 					     extack);
395 		if (err)
396 			return err;
397 	}
398 
399 	if (tb[IFLA_BRIDGE_MRP_RING_STATE]) {
400 		err = br_mrp_ring_state_parse(br,
401 					      tb[IFLA_BRIDGE_MRP_RING_STATE],
402 					      extack);
403 		if (err)
404 			return err;
405 	}
406 
407 	if (tb[IFLA_BRIDGE_MRP_RING_ROLE]) {
408 		err = br_mrp_ring_role_parse(br, tb[IFLA_BRIDGE_MRP_RING_ROLE],
409 					     extack);
410 		if (err)
411 			return err;
412 	}
413 
414 	if (tb[IFLA_BRIDGE_MRP_START_TEST]) {
415 		err = br_mrp_start_test_parse(br,
416 					      tb[IFLA_BRIDGE_MRP_START_TEST],
417 					      extack);
418 		if (err)
419 			return err;
420 	}
421 
422 	if (tb[IFLA_BRIDGE_MRP_IN_STATE]) {
423 		err = br_mrp_in_state_parse(br, tb[IFLA_BRIDGE_MRP_IN_STATE],
424 					    extack);
425 		if (err)
426 			return err;
427 	}
428 
429 	if (tb[IFLA_BRIDGE_MRP_IN_ROLE]) {
430 		err = br_mrp_in_role_parse(br, tb[IFLA_BRIDGE_MRP_IN_ROLE],
431 					   extack);
432 		if (err)
433 			return err;
434 	}
435 
436 	if (tb[IFLA_BRIDGE_MRP_START_IN_TEST]) {
437 		err = br_mrp_start_in_test_parse(br,
438 						 tb[IFLA_BRIDGE_MRP_START_IN_TEST],
439 						 extack);
440 		if (err)
441 			return err;
442 	}
443 
444 	return 0;
445 }
446 
447 int br_mrp_fill_info(struct sk_buff *skb, struct net_bridge *br)
448 {
449 	struct nlattr *tb, *mrp_tb;
450 	struct br_mrp *mrp;
451 
452 	mrp_tb = nla_nest_start_noflag(skb, IFLA_BRIDGE_MRP);
453 	if (!mrp_tb)
454 		return -EMSGSIZE;
455 
456 	hlist_for_each_entry_rcu(mrp, &br->mrp_list, list) {
457 		struct net_bridge_port *p;
458 
459 		tb = nla_nest_start_noflag(skb, IFLA_BRIDGE_MRP_INFO);
460 		if (!tb)
461 			goto nla_info_failure;
462 
463 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_RING_ID,
464 				mrp->ring_id))
465 			goto nla_put_failure;
466 
467 		p = rcu_dereference(mrp->p_port);
468 		if (p && nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_P_IFINDEX,
469 				     p->dev->ifindex))
470 			goto nla_put_failure;
471 
472 		p = rcu_dereference(mrp->s_port);
473 		if (p && nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_S_IFINDEX,
474 				     p->dev->ifindex))
475 			goto nla_put_failure;
476 
477 		p = rcu_dereference(mrp->i_port);
478 		if (p && nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_I_IFINDEX,
479 				     p->dev->ifindex))
480 			goto nla_put_failure;
481 
482 		if (nla_put_u16(skb, IFLA_BRIDGE_MRP_INFO_PRIO,
483 				mrp->prio))
484 			goto nla_put_failure;
485 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_RING_STATE,
486 				mrp->ring_state))
487 			goto nla_put_failure;
488 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_RING_ROLE,
489 				mrp->ring_role))
490 			goto nla_put_failure;
491 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_TEST_INTERVAL,
492 				mrp->test_interval))
493 			goto nla_put_failure;
494 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_TEST_MAX_MISS,
495 				mrp->test_max_miss))
496 			goto nla_put_failure;
497 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_TEST_MONITOR,
498 				mrp->test_monitor))
499 			goto nla_put_failure;
500 
501 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_IN_STATE,
502 				mrp->in_state))
503 			goto nla_put_failure;
504 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_IN_ROLE,
505 				mrp->in_role))
506 			goto nla_put_failure;
507 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_IN_TEST_INTERVAL,
508 				mrp->in_test_interval))
509 			goto nla_put_failure;
510 		if (nla_put_u32(skb, IFLA_BRIDGE_MRP_INFO_IN_TEST_MAX_MISS,
511 				mrp->in_test_max_miss))
512 			goto nla_put_failure;
513 
514 		nla_nest_end(skb, tb);
515 	}
516 	nla_nest_end(skb, mrp_tb);
517 
518 	return 0;
519 
520 nla_put_failure:
521 	nla_nest_cancel(skb, tb);
522 
523 nla_info_failure:
524 	nla_nest_cancel(skb, mrp_tb);
525 
526 	return -EMSGSIZE;
527 }
528 
529 int br_mrp_ring_port_open(struct net_device *dev, u8 loc)
530 {
531 	struct net_bridge_port *p;
532 	int err = 0;
533 
534 	p = br_port_get_rcu(dev);
535 	if (!p) {
536 		err = -EINVAL;
537 		goto out;
538 	}
539 
540 	if (loc)
541 		p->flags |= BR_MRP_LOST_CONT;
542 	else
543 		p->flags &= ~BR_MRP_LOST_CONT;
544 
545 	br_ifinfo_notify(RTM_NEWLINK, NULL, p);
546 
547 out:
548 	return err;
549 }
550 
551 int br_mrp_in_port_open(struct net_device *dev, u8 loc)
552 {
553 	struct net_bridge_port *p;
554 	int err = 0;
555 
556 	p = br_port_get_rcu(dev);
557 	if (!p) {
558 		err = -EINVAL;
559 		goto out;
560 	}
561 
562 	if (loc)
563 		p->flags |= BR_MRP_LOST_IN_CONT;
564 	else
565 		p->flags &= ~BR_MRP_LOST_IN_CONT;
566 
567 	br_ifinfo_notify(RTM_NEWLINK, NULL, p);
568 
569 out:
570 	return err;
571 }
572