1 /* SPDX-License-Identifier: LGPL-2.1+ */
3 Copyright © 2014 Tom Gundersen <teg@jklm.no>
4 Copyright © 2014 Susant Sahani
10 #include "netlink-util.h"
11 #include "netdev/bridge.h"
12 #include "networkd-manager.h"
13 #include "vlan-util.h"
15 /* callback for brige netdev's parameter set */
16 static int netdev_bridge_set_handler(sd_netlink
*rtnl
, sd_netlink_message
*m
, void *userdata
) {
17 _cleanup_(netdev_unrefp
) NetDev
*netdev
= userdata
;
23 r
= sd_netlink_message_get_errno(m
);
25 log_netdev_warning_errno(netdev
, r
, "Bridge parameters could not be set: %m");
29 log_netdev_debug(netdev
, "Bridge parameters set success");
34 static int netdev_bridge_post_create(NetDev
*netdev
, Link
*link
, sd_netlink_message
*m
) {
35 _cleanup_(sd_netlink_message_unrefp
) sd_netlink_message
*req
= NULL
;
45 r
= sd_rtnl_message_new_link(netdev
->manager
->rtnl
, &req
, RTM_NEWLINK
, netdev
->ifindex
);
47 return log_netdev_error_errno(netdev
, r
, "Could not allocate RTM_SETLINK message: %m");
49 r
= sd_netlink_message_set_flags(req
, NLM_F_REQUEST
| NLM_F_ACK
);
51 return log_link_error_errno(link
, r
, "Could not set netlink flags: %m");
53 r
= sd_netlink_message_open_container(req
, IFLA_LINKINFO
);
55 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_PROTINFO attribute: %m");
57 r
= sd_netlink_message_open_container_union(req
, IFLA_INFO_DATA
, netdev_kind_to_string(netdev
->kind
));
59 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
61 /* convert to jiffes */
62 if (b
->forward_delay
!= USEC_INFINITY
) {
63 r
= sd_netlink_message_append_u32(req
, IFLA_BR_FORWARD_DELAY
, usec_to_jiffies(b
->forward_delay
));
65 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_FORWARD_DELAY attribute: %m");
68 if (b
->hello_time
> 0) {
69 r
= sd_netlink_message_append_u32(req
, IFLA_BR_HELLO_TIME
, usec_to_jiffies(b
->hello_time
));
71 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_HELLO_TIME attribute: %m");
75 r
= sd_netlink_message_append_u32(req
, IFLA_BR_MAX_AGE
, usec_to_jiffies(b
->max_age
));
77 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_MAX_AGE attribute: %m");
80 if (b
->ageing_time
!= USEC_INFINITY
) {
81 r
= sd_netlink_message_append_u32(req
, IFLA_BR_AGEING_TIME
, usec_to_jiffies(b
->ageing_time
));
83 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_AGEING_TIME attribute: %m");
86 if (b
->priority
> 0) {
87 r
= sd_netlink_message_append_u16(req
, IFLA_BR_PRIORITY
, b
->priority
);
89 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_PRIORITY attribute: %m");
92 if (b
->group_fwd_mask
> 0) {
93 r
= sd_netlink_message_append_u16(req
, IFLA_BR_GROUP_FWD_MASK
, b
->group_fwd_mask
);
95 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_GROUP_FWD_MASK attribute: %m");
98 if (b
->default_pvid
!= VLANID_INVALID
) {
99 r
= sd_netlink_message_append_u16(req
, IFLA_BR_VLAN_DEFAULT_PVID
, b
->default_pvid
);
101 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_VLAN_DEFAULT_PVID attribute: %m");
104 if (b
->mcast_querier
>= 0) {
105 r
= sd_netlink_message_append_u8(req
, IFLA_BR_MCAST_QUERIER
, b
->mcast_querier
);
107 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_MCAST_QUERIER attribute: %m");
110 if (b
->mcast_snooping
>= 0) {
111 r
= sd_netlink_message_append_u8(req
, IFLA_BR_MCAST_SNOOPING
, b
->mcast_snooping
);
113 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_MCAST_SNOOPING attribute: %m");
116 if (b
->vlan_filtering
>= 0) {
117 r
= sd_netlink_message_append_u8(req
, IFLA_BR_VLAN_FILTERING
, b
->vlan_filtering
);
119 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_VLAN_FILTERING attribute: %m");
123 r
= sd_netlink_message_append_u32(req
, IFLA_BR_STP_STATE
, b
->stp
);
125 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_BR_STP_STATE attribute: %m");
128 r
= sd_netlink_message_close_container(req
);
130 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
132 r
= sd_netlink_message_close_container(req
);
134 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
136 r
= sd_netlink_call_async(netdev
->manager
->rtnl
, req
, netdev_bridge_set_handler
, netdev
, 0, NULL
);
138 return log_netdev_error_errno(netdev
, r
, "Could not send rtnetlink message: %m");
145 static void bridge_init(NetDev
*n
) {
152 b
->mcast_querier
= -1;
153 b
->mcast_snooping
= -1;
154 b
->vlan_filtering
= -1;
156 b
->default_pvid
= VLANID_INVALID
;
157 b
->forward_delay
= USEC_INFINITY
;
158 b
->ageing_time
= USEC_INFINITY
;
161 const NetDevVTable bridge_vtable
= {
162 .object_size
= sizeof(Bridge
),
164 .sections
= "Match\0NetDev\0Bridge\0",
165 .post_create
= netdev_bridge_post_create
,
166 .create_type
= NETDEV_CREATE_MASTER
,