diff options
| author | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2015-04-14 11:51:33 -0400 |
|---|---|---|
| committer | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2015-04-14 11:51:33 -0400 |
| commit | 85a3685852d9ac7d92be9d824533c915a4597fa4 (patch) | |
| tree | b7c542e2061cf96c9f7ad500fa12567f9ff0b39f /net/switchdev/switchdev.c | |
| parent | 92bac83dd79e60e65c475222e41a992a70434beb (diff) | |
| parent | 8b8a518ef16be2de27207991e32fc32b0475c767 (diff) | |
Merge branch 'next' into for-linus
Prepare first round of input updates for 4.1 merge window.
Diffstat (limited to 'net/switchdev/switchdev.c')
| -rw-r--r-- | net/switchdev/switchdev.c | 175 |
1 files changed, 175 insertions, 0 deletions
diff --git a/net/switchdev/switchdev.c b/net/switchdev/switchdev.c index d162b21b14bd..8c1e558db118 100644 --- a/net/switchdev/switchdev.c +++ b/net/switchdev/switchdev.c | |||
| @@ -11,6 +11,8 @@ | |||
| 11 | #include <linux/kernel.h> | 11 | #include <linux/kernel.h> |
| 12 | #include <linux/types.h> | 12 | #include <linux/types.h> |
| 13 | #include <linux/init.h> | 13 | #include <linux/init.h> |
| 14 | #include <linux/mutex.h> | ||
| 15 | #include <linux/notifier.h> | ||
| 14 | #include <linux/netdevice.h> | 16 | #include <linux/netdevice.h> |
| 15 | #include <net/switchdev.h> | 17 | #include <net/switchdev.h> |
| 16 | 18 | ||
| @@ -50,3 +52,176 @@ int netdev_switch_port_stp_update(struct net_device *dev, u8 state) | |||
| 50 | return ops->ndo_switch_port_stp_update(dev, state); | 52 | return ops->ndo_switch_port_stp_update(dev, state); |
| 51 | } | 53 | } |
| 52 | EXPORT_SYMBOL(netdev_switch_port_stp_update); | 54 | EXPORT_SYMBOL(netdev_switch_port_stp_update); |
| 55 | |||
| 56 | static DEFINE_MUTEX(netdev_switch_mutex); | ||
| 57 | static RAW_NOTIFIER_HEAD(netdev_switch_notif_chain); | ||
| 58 | |||
| 59 | /** | ||
| 60 | * register_netdev_switch_notifier - Register nofifier | ||
| 61 | * @nb: notifier_block | ||
| 62 | * | ||
| 63 | * Register switch device notifier. This should be used by code | ||
| 64 | * which needs to monitor events happening in particular device. | ||
| 65 | * Return values are same as for atomic_notifier_chain_register(). | ||
| 66 | */ | ||
| 67 | int register_netdev_switch_notifier(struct notifier_block *nb) | ||
| 68 | { | ||
| 69 | int err; | ||
| 70 | |||
| 71 | mutex_lock(&netdev_switch_mutex); | ||
| 72 | err = raw_notifier_chain_register(&netdev_switch_notif_chain, nb); | ||
| 73 | mutex_unlock(&netdev_switch_mutex); | ||
| 74 | return err; | ||
| 75 | } | ||
| 76 | EXPORT_SYMBOL(register_netdev_switch_notifier); | ||
| 77 | |||
| 78 | /** | ||
| 79 | * unregister_netdev_switch_notifier - Unregister nofifier | ||
| 80 | * @nb: notifier_block | ||
| 81 | * | ||
| 82 | * Unregister switch device notifier. | ||
| 83 | * Return values are same as for atomic_notifier_chain_unregister(). | ||
| 84 | */ | ||
| 85 | int unregister_netdev_switch_notifier(struct notifier_block *nb) | ||
| 86 | { | ||
| 87 | int err; | ||
| 88 | |||
| 89 | mutex_lock(&netdev_switch_mutex); | ||
| 90 | err = raw_notifier_chain_unregister(&netdev_switch_notif_chain, nb); | ||
| 91 | mutex_unlock(&netdev_switch_mutex); | ||
| 92 | return err; | ||
| 93 | } | ||
| 94 | EXPORT_SYMBOL(unregister_netdev_switch_notifier); | ||
| 95 | |||
| 96 | /** | ||
| 97 | * call_netdev_switch_notifiers - Call nofifiers | ||
| 98 | * @val: value passed unmodified to notifier function | ||
| 99 | * @dev: port device | ||
| 100 | * @info: notifier information data | ||
| 101 | * | ||
| 102 | * Call all network notifier blocks. This should be called by driver | ||
| 103 | * when it needs to propagate hardware event. | ||
| 104 | * Return values are same as for atomic_notifier_call_chain(). | ||
| 105 | */ | ||
| 106 | int call_netdev_switch_notifiers(unsigned long val, struct net_device *dev, | ||
| 107 | struct netdev_switch_notifier_info *info) | ||
| 108 | { | ||
| 109 | int err; | ||
| 110 | |||
| 111 | info->dev = dev; | ||
| 112 | mutex_lock(&netdev_switch_mutex); | ||
| 113 | err = raw_notifier_call_chain(&netdev_switch_notif_chain, val, info); | ||
| 114 | mutex_unlock(&netdev_switch_mutex); | ||
| 115 | return err; | ||
| 116 | } | ||
| 117 | EXPORT_SYMBOL(call_netdev_switch_notifiers); | ||
| 118 | |||
| 119 | /** | ||
| 120 | * netdev_switch_port_bridge_setlink - Notify switch device port of bridge | ||
| 121 | * port attributes | ||
| 122 | * | ||
| 123 | * @dev: port device | ||
| 124 | * @nlh: netlink msg with bridge port attributes | ||
| 125 | * @flags: bridge setlink flags | ||
| 126 | * | ||
| 127 | * Notify switch device port of bridge port attributes | ||
| 128 | */ | ||
| 129 | int netdev_switch_port_bridge_setlink(struct net_device *dev, | ||
| 130 | struct nlmsghdr *nlh, u16 flags) | ||
| 131 | { | ||
| 132 | const struct net_device_ops *ops = dev->netdev_ops; | ||
| 133 | |||
| 134 | if (!(dev->features & NETIF_F_HW_SWITCH_OFFLOAD)) | ||
| 135 | return 0; | ||
| 136 | |||
| 137 | if (!ops->ndo_bridge_setlink) | ||
| 138 | return -EOPNOTSUPP; | ||
| 139 | |||
| 140 | return ops->ndo_bridge_setlink(dev, nlh, flags); | ||
| 141 | } | ||
| 142 | EXPORT_SYMBOL(netdev_switch_port_bridge_setlink); | ||
| 143 | |||
| 144 | /** | ||
| 145 | * netdev_switch_port_bridge_dellink - Notify switch device port of bridge | ||
| 146 | * port attribute delete | ||
| 147 | * | ||
| 148 | * @dev: port device | ||
| 149 | * @nlh: netlink msg with bridge port attributes | ||
| 150 | * @flags: bridge setlink flags | ||
| 151 | * | ||
| 152 | * Notify switch device port of bridge port attribute delete | ||
| 153 | */ | ||
| 154 | int netdev_switch_port_bridge_dellink(struct net_device *dev, | ||
| 155 | struct nlmsghdr *nlh, u16 flags) | ||
| 156 | { | ||
| 157 | const struct net_device_ops *ops = dev->netdev_ops; | ||
| 158 | |||
| 159 | if (!(dev->features & NETIF_F_HW_SWITCH_OFFLOAD)) | ||
| 160 | return 0; | ||
| 161 | |||
| 162 | if (!ops->ndo_bridge_dellink) | ||
| 163 | return -EOPNOTSUPP; | ||
| 164 | |||
| 165 | return ops->ndo_bridge_dellink(dev, nlh, flags); | ||
| 166 | } | ||
| 167 | EXPORT_SYMBOL(netdev_switch_port_bridge_dellink); | ||
| 168 | |||
| 169 | /** | ||
| 170 | * ndo_dflt_netdev_switch_port_bridge_setlink - default ndo bridge setlink | ||
| 171 | * op for master devices | ||
| 172 | * | ||
| 173 | * @dev: port device | ||
| 174 | * @nlh: netlink msg with bridge port attributes | ||
| 175 | * @flags: bridge setlink flags | ||
| 176 | * | ||
| 177 | * Notify master device slaves of bridge port attributes | ||
| 178 | */ | ||
| 179 | int ndo_dflt_netdev_switch_port_bridge_setlink(struct net_device *dev, | ||
| 180 | struct nlmsghdr *nlh, u16 flags) | ||
| 181 | { | ||
| 182 | struct net_device *lower_dev; | ||
| 183 | struct list_head *iter; | ||
| 184 | int ret = 0, err = 0; | ||
| 185 | |||
| 186 | if (!(dev->features & NETIF_F_HW_SWITCH_OFFLOAD)) | ||
| 187 | return ret; | ||
| 188 | |||
| 189 | netdev_for_each_lower_dev(dev, lower_dev, iter) { | ||
| 190 | err = netdev_switch_port_bridge_setlink(lower_dev, nlh, flags); | ||
| 191 | if (err && err != -EOPNOTSUPP) | ||
| 192 | ret = err; | ||
| 193 | } | ||
| 194 | |||
| 195 | return ret; | ||
| 196 | } | ||
| 197 | EXPORT_SYMBOL(ndo_dflt_netdev_switch_port_bridge_setlink); | ||
| 198 | |||
| 199 | /** | ||
| 200 | * ndo_dflt_netdev_switch_port_bridge_dellink - default ndo bridge dellink | ||
| 201 | * op for master devices | ||
| 202 | * | ||
| 203 | * @dev: port device | ||
| 204 | * @nlh: netlink msg with bridge port attributes | ||
| 205 | * @flags: bridge dellink flags | ||
| 206 | * | ||
| 207 | * Notify master device slaves of bridge port attribute deletes | ||
| 208 | */ | ||
| 209 | int ndo_dflt_netdev_switch_port_bridge_dellink(struct net_device *dev, | ||
| 210 | struct nlmsghdr *nlh, u16 flags) | ||
| 211 | { | ||
| 212 | struct net_device *lower_dev; | ||
| 213 | struct list_head *iter; | ||
| 214 | int ret = 0, err = 0; | ||
| 215 | |||
| 216 | if (!(dev->features & NETIF_F_HW_SWITCH_OFFLOAD)) | ||
| 217 | return ret; | ||
| 218 | |||
| 219 | netdev_for_each_lower_dev(dev, lower_dev, iter) { | ||
| 220 | err = netdev_switch_port_bridge_dellink(lower_dev, nlh, flags); | ||
| 221 | if (err && err != -EOPNOTSUPP) | ||
| 222 | ret = err; | ||
| 223 | } | ||
| 224 | |||
| 225 | return ret; | ||
| 226 | } | ||
| 227 | EXPORT_SYMBOL(ndo_dflt_netdev_switch_port_bridge_dellink); | ||
