aboutsummaryrefslogtreecommitdiffstats
path: root/net/switchdev/switchdev.c
diff options
context:
space:
mode:
authorScott Feldman <sfeldma@gmail.com>2015-05-10 12:47:52 -0400
committerDavid S. Miller <davem@davemloft.net>2015-05-12 18:43:53 -0400
commit491d0f1533ac750260406dbf84cdad44fd3d8a29 (patch)
tree6443f2004cf3198a43418f6c7f0faa33f2e23052 /net/switchdev/switchdev.c
parent3563606258cf3b8f02eabddb1cb45a94c44d9611 (diff)
switchdev: introduce switchdev add/del obj ops
Like switchdev attr get/set, add new switchdev obj add/del. switchdev objs will be things like VLANs or FIB entries, so add/del fits better for objects than get/set used for attributes. Use same two-phase prepare-commit transaction model as in attr set. Signed-off-by: Scott Feldman <sfeldma@gmail.com> Acked-by: Sridhar Samudrala <sridhar.samudrala@intel.com> Acked-by: Jiri Pirko <jiri@resnulli.us> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/switchdev/switchdev.c')
-rw-r--r--net/switchdev/switchdev.c107
1 files changed, 107 insertions, 0 deletions
diff --git a/net/switchdev/switchdev.c b/net/switchdev/switchdev.c
index a3c359004902..3d4d99a70b80 100644
--- a/net/switchdev/switchdev.c
+++ b/net/switchdev/switchdev.c
@@ -187,6 +187,113 @@ int switchdev_port_attr_set(struct net_device *dev, struct switchdev_attr *attr)
187} 187}
188EXPORT_SYMBOL_GPL(switchdev_port_attr_set); 188EXPORT_SYMBOL_GPL(switchdev_port_attr_set);
189 189
190int __switchdev_port_obj_add(struct net_device *dev, struct switchdev_obj *obj)
191{
192 const struct switchdev_ops *ops = dev->switchdev_ops;
193 struct net_device *lower_dev;
194 struct list_head *iter;
195 int err = -EOPNOTSUPP;
196
197 if (ops && ops->switchdev_port_obj_add)
198 return ops->switchdev_port_obj_add(dev, obj);
199
200 /* Switch device port(s) may be stacked under
201 * bond/team/vlan dev, so recurse down to add object on
202 * each port.
203 */
204
205 netdev_for_each_lower_dev(dev, lower_dev, iter) {
206 err = __switchdev_port_obj_add(lower_dev, obj);
207 if (err)
208 break;
209 }
210
211 return err;
212}
213
214/**
215 * switchdev_port_obj_add - Add port object
216 *
217 * @dev: port device
218 * @obj: object to add
219 *
220 * Use a 2-phase prepare-commit transaction model to ensure
221 * system is not left in a partially updated state due to
222 * failure from driver/device.
223 *
224 * rtnl_lock must be held.
225 */
226int switchdev_port_obj_add(struct net_device *dev, struct switchdev_obj *obj)
227{
228 int err;
229
230 ASSERT_RTNL();
231
232 /* Phase I: prepare for obj add. Driver/device should fail
233 * here if there are going to be issues in the commit phase,
234 * such as lack of resources or support. The driver/device
235 * should reserve resources needed for the commit phase here,
236 * but should not commit the obj.
237 */
238
239 obj->trans = SWITCHDEV_TRANS_PREPARE;
240 err = __switchdev_port_obj_add(dev, obj);
241 if (err) {
242 /* Prepare phase failed: abort the transaction. Any
243 * resources reserved in the prepare phase are
244 * released.
245 */
246
247 obj->trans = SWITCHDEV_TRANS_ABORT;
248 __switchdev_port_obj_add(dev, obj);
249
250 return err;
251 }
252
253 /* Phase II: commit obj add. This cannot fail as a fault
254 * of driver/device. If it does, it's a bug in the driver/device
255 * because the driver said everythings was OK in phase I.
256 */
257
258 obj->trans = SWITCHDEV_TRANS_COMMIT;
259 err = __switchdev_port_obj_add(dev, obj);
260 WARN(err, "%s: Commit of object (id=%d) failed.\n", dev->name, obj->id);
261
262 return err;
263}
264EXPORT_SYMBOL_GPL(switchdev_port_obj_add);
265
266/**
267 * switchdev_port_obj_del - Delete port object
268 *
269 * @dev: port device
270 * @obj: object to delete
271 */
272int switchdev_port_obj_del(struct net_device *dev, struct switchdev_obj *obj)
273{
274 const struct switchdev_ops *ops = dev->switchdev_ops;
275 struct net_device *lower_dev;
276 struct list_head *iter;
277 int err = -EOPNOTSUPP;
278
279 if (ops && ops->switchdev_port_obj_del)
280 return ops->switchdev_port_obj_del(dev, obj);
281
282 /* Switch device port(s) may be stacked under
283 * bond/team/vlan dev, so recurse down to delete object on
284 * each port.
285 */
286
287 netdev_for_each_lower_dev(dev, lower_dev, iter) {
288 err = switchdev_port_obj_del(lower_dev, obj);
289 if (err)
290 break;
291 }
292
293 return err;
294}
295EXPORT_SYMBOL_GPL(switchdev_port_obj_del);
296
190static DEFINE_MUTEX(switchdev_mutex); 297static DEFINE_MUTEX(switchdev_mutex);
191static RAW_NOTIFIER_HEAD(switchdev_notif_chain); 298static RAW_NOTIFIER_HEAD(switchdev_notif_chain);
192 299