123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107 |
- // SPDX-License-Identifier: GPL-2.0
- /* Copyright(c) 2021 Intel Corporation. All rights rsvd. */
- #include <linux/init.h>
- #include <linux/kernel.h>
- #include <linux/module.h>
- #include <linux/device.h>
- #include <linux/device/bus.h>
- #include "idxd.h"
- extern int device_driver_attach(struct device_driver *drv, struct device *dev);
- extern void device_driver_detach(struct device *dev);
- #define DRIVER_ATTR_IGNORE_LOCKDEP(_name, _mode, _show, _store) \
- struct driver_attribute driver_attr_##_name = \
- __ATTR_IGNORE_LOCKDEP(_name, _mode, _show, _store)
- static ssize_t unbind_store(struct device_driver *drv, const char *buf, size_t count)
- {
- struct bus_type *bus = drv->bus;
- struct device *dev;
- int rc = -ENODEV;
- dev = bus_find_device_by_name(bus, NULL, buf);
- if (dev && dev->driver) {
- device_driver_detach(dev);
- rc = count;
- }
- return rc;
- }
- static DRIVER_ATTR_IGNORE_LOCKDEP(unbind, 0200, NULL, unbind_store);
- static ssize_t bind_store(struct device_driver *drv, const char *buf, size_t count)
- {
- struct bus_type *bus = drv->bus;
- struct device *dev;
- struct device_driver *alt_drv = NULL;
- int rc = -ENODEV;
- struct idxd_dev *idxd_dev;
- dev = bus_find_device_by_name(bus, NULL, buf);
- if (!dev || dev->driver || drv != &dsa_drv.drv)
- return -ENODEV;
- idxd_dev = confdev_to_idxd_dev(dev);
- if (is_idxd_dev(idxd_dev)) {
- alt_drv = driver_find("idxd", bus);
- } else if (is_idxd_wq_dev(idxd_dev)) {
- struct idxd_wq *wq = confdev_to_wq(dev);
- if (is_idxd_wq_kernel(wq))
- alt_drv = driver_find("dmaengine", bus);
- else if (is_idxd_wq_user(wq))
- alt_drv = driver_find("user", bus);
- }
- if (!alt_drv)
- return -ENODEV;
- rc = device_driver_attach(alt_drv, dev);
- if (rc < 0)
- return rc;
- return count;
- }
- static DRIVER_ATTR_IGNORE_LOCKDEP(bind, 0200, NULL, bind_store);
- static struct attribute *dsa_drv_compat_attrs[] = {
- &driver_attr_bind.attr,
- &driver_attr_unbind.attr,
- NULL,
- };
- static const struct attribute_group dsa_drv_compat_attr_group = {
- .attrs = dsa_drv_compat_attrs,
- };
- static const struct attribute_group *dsa_drv_compat_groups[] = {
- &dsa_drv_compat_attr_group,
- NULL,
- };
- static int idxd_dsa_drv_probe(struct idxd_dev *idxd_dev)
- {
- return -ENODEV;
- }
- static void idxd_dsa_drv_remove(struct idxd_dev *idxd_dev)
- {
- }
- static enum idxd_dev_type dev_types[] = {
- IDXD_DEV_NONE,
- };
- struct idxd_device_driver dsa_drv = {
- .name = "dsa",
- .probe = idxd_dsa_drv_probe,
- .remove = idxd_dsa_drv_remove,
- .type = dev_types,
- .drv = {
- .suppress_bind_attrs = true,
- .groups = dsa_drv_compat_groups,
- },
- };
- module_idxd_driver(dsa_drv);
- MODULE_IMPORT_NS(IDXD);
|