ipack: save carrier owner to allow device to get it
authorFederico Vaga <federico.vaga@cern.ch>
Tue, 2 Sep 2014 15:31:40 +0000 (17:31 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 24 Sep 2014 06:13:13 +0000 (23:13 -0700)
There was not any kind of protection against carrier driver removal.
In this way, device driver can 'get' the carrier driver when it is
using it.

Signed-off-by: Federico Vaga <federico.vaga@cern.ch>
Acked-by: Samuel Iglesias Gonsalvez <siglesias@igalia.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/ipack/carriers/tpci200.c
drivers/ipack/ipack.c
include/linux/ipack.h

index de5e32151a1e1c522ced94bea4ed0e67157d6def..9b23843dcad4d7bb24851f30f663da84b503310e 100644 (file)
@@ -572,7 +572,8 @@ static int tpci200_pci_probe(struct pci_dev *pdev,
        /* Register the carrier in the industry pack bus driver */
        tpci200->info->ipack_bus = ipack_bus_register(&pdev->dev,
                                                      TPCI200_NB_SLOT,
-                                                     &tpci200_bus_ops);
+                                                     &tpci200_bus_ops,
+                                                     THIS_MODULE);
        if (!tpci200->info->ipack_bus) {
                dev_err(&pdev->dev,
                        "error registering the carrier on ipack driver\n");
index d0016ba469edeed10c58fd1ce93c5615dc2735c4..c0e7b624ce5475aed0022d3b6f79a043110740ff 100644 (file)
@@ -206,7 +206,8 @@ static struct bus_type ipack_bus_type = {
 };
 
 struct ipack_bus_device *ipack_bus_register(struct device *parent, int slots,
-                                           const struct ipack_bus_ops *ops)
+                                           const struct ipack_bus_ops *ops,
+                                           struct module *owner)
 {
        int bus_nr;
        struct ipack_bus_device *bus;
@@ -225,6 +226,7 @@ struct ipack_bus_device *ipack_bus_register(struct device *parent, int slots,
        bus->parent = parent;
        bus->slots = slots;
        bus->ops = ops;
+       bus->owner = owner;
        return bus;
 }
 EXPORT_SYMBOL_GPL(ipack_bus_register);
index 1888e06ddf64d6f08824c9c4f842f9c884e1e1a4..8bddc3fbdddfb164cb4b8dc671311b92441b7460 100644 (file)
@@ -172,6 +172,7 @@ struct ipack_bus_ops {
  *     @ops: bus operations for the mezzanine drivers
  */
 struct ipack_bus_device {
+       struct module *owner;
        struct device *parent;
        int slots;
        int bus_nr;
@@ -189,7 +190,8 @@ struct ipack_bus_device {
  * available bus device in ipack.
  */
 struct ipack_bus_device *ipack_bus_register(struct device *parent, int slots,
-                                           const struct ipack_bus_ops *ops);
+                                           const struct ipack_bus_ops *ops,
+                                           struct module *owner);
 
 /**
  *     ipack_bus_unregister -- unregister an ipack bus
@@ -265,3 +267,23 @@ void ipack_put_device(struct ipack_device *dev);
         .format = (_format), \
         .vendor = (vend), \
         .device = (dev)
+
+/**
+ * ipack_get_carrier - it increase the carrier ref. counter of
+ *                     the carrier module
+ * @dev: mezzanine device which wants to get the carrier
+ */
+static inline int ipack_get_carrier(struct ipack_device *dev)
+{
+       return try_module_get(dev->bus->owner);
+}
+
+/**
+ * ipack_get_carrier - it decrease the carrier ref. counter of
+ *                     the carrier module
+ * @dev: mezzanine device which wants to get the carrier
+ */
+static inline void ipack_put_carrier(struct ipack_device *dev)
+{
+       module_put(dev->bus->owner);
+}