162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* Copyright(c) 2021 Intel Corporation. All rights rsvd. */ 362306a36Sopenharmony_ci#include <linux/init.h> 462306a36Sopenharmony_ci#include <linux/kernel.h> 562306a36Sopenharmony_ci#include <linux/module.h> 662306a36Sopenharmony_ci#include <linux/device.h> 762306a36Sopenharmony_ci#include <linux/device/bus.h> 862306a36Sopenharmony_ci#include "idxd.h" 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ciextern int device_driver_attach(struct device_driver *drv, struct device *dev); 1162306a36Sopenharmony_ciextern void device_driver_detach(struct device *dev); 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#define DRIVER_ATTR_IGNORE_LOCKDEP(_name, _mode, _show, _store) \ 1462306a36Sopenharmony_ci struct driver_attribute driver_attr_##_name = \ 1562306a36Sopenharmony_ci __ATTR_IGNORE_LOCKDEP(_name, _mode, _show, _store) 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_cistatic ssize_t unbind_store(struct device_driver *drv, const char *buf, size_t count) 1862306a36Sopenharmony_ci{ 1962306a36Sopenharmony_ci const struct bus_type *bus = drv->bus; 2062306a36Sopenharmony_ci struct device *dev; 2162306a36Sopenharmony_ci int rc = -ENODEV; 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci dev = bus_find_device_by_name(bus, NULL, buf); 2462306a36Sopenharmony_ci if (dev && dev->driver) { 2562306a36Sopenharmony_ci device_driver_detach(dev); 2662306a36Sopenharmony_ci rc = count; 2762306a36Sopenharmony_ci } 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci return rc; 3062306a36Sopenharmony_ci} 3162306a36Sopenharmony_cistatic DRIVER_ATTR_IGNORE_LOCKDEP(unbind, 0200, NULL, unbind_store); 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_cistatic ssize_t bind_store(struct device_driver *drv, const char *buf, size_t count) 3462306a36Sopenharmony_ci{ 3562306a36Sopenharmony_ci const struct bus_type *bus = drv->bus; 3662306a36Sopenharmony_ci struct device *dev; 3762306a36Sopenharmony_ci struct device_driver *alt_drv = NULL; 3862306a36Sopenharmony_ci int rc = -ENODEV; 3962306a36Sopenharmony_ci struct idxd_dev *idxd_dev; 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_ci dev = bus_find_device_by_name(bus, NULL, buf); 4262306a36Sopenharmony_ci if (!dev || dev->driver || drv != &dsa_drv.drv) 4362306a36Sopenharmony_ci return -ENODEV; 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_ci idxd_dev = confdev_to_idxd_dev(dev); 4662306a36Sopenharmony_ci if (is_idxd_dev(idxd_dev)) { 4762306a36Sopenharmony_ci alt_drv = driver_find("idxd", bus); 4862306a36Sopenharmony_ci } else if (is_idxd_wq_dev(idxd_dev)) { 4962306a36Sopenharmony_ci struct idxd_wq *wq = confdev_to_wq(dev); 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci if (is_idxd_wq_kernel(wq)) 5262306a36Sopenharmony_ci alt_drv = driver_find("dmaengine", bus); 5362306a36Sopenharmony_ci else if (is_idxd_wq_user(wq)) 5462306a36Sopenharmony_ci alt_drv = driver_find("user", bus); 5562306a36Sopenharmony_ci } 5662306a36Sopenharmony_ci if (!alt_drv) 5762306a36Sopenharmony_ci return -ENODEV; 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci rc = device_driver_attach(alt_drv, dev); 6062306a36Sopenharmony_ci if (rc < 0) 6162306a36Sopenharmony_ci return rc; 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci return count; 6462306a36Sopenharmony_ci} 6562306a36Sopenharmony_cistatic DRIVER_ATTR_IGNORE_LOCKDEP(bind, 0200, NULL, bind_store); 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_cistatic struct attribute *dsa_drv_compat_attrs[] = { 6862306a36Sopenharmony_ci &driver_attr_bind.attr, 6962306a36Sopenharmony_ci &driver_attr_unbind.attr, 7062306a36Sopenharmony_ci NULL, 7162306a36Sopenharmony_ci}; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_cistatic const struct attribute_group dsa_drv_compat_attr_group = { 7462306a36Sopenharmony_ci .attrs = dsa_drv_compat_attrs, 7562306a36Sopenharmony_ci}; 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_cistatic const struct attribute_group *dsa_drv_compat_groups[] = { 7862306a36Sopenharmony_ci &dsa_drv_compat_attr_group, 7962306a36Sopenharmony_ci NULL, 8062306a36Sopenharmony_ci}; 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_cistatic int idxd_dsa_drv_probe(struct idxd_dev *idxd_dev) 8362306a36Sopenharmony_ci{ 8462306a36Sopenharmony_ci return -ENODEV; 8562306a36Sopenharmony_ci} 8662306a36Sopenharmony_ci 8762306a36Sopenharmony_cistatic void idxd_dsa_drv_remove(struct idxd_dev *idxd_dev) 8862306a36Sopenharmony_ci{ 8962306a36Sopenharmony_ci} 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cistatic enum idxd_dev_type dev_types[] = { 9262306a36Sopenharmony_ci IDXD_DEV_NONE, 9362306a36Sopenharmony_ci}; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_cistruct idxd_device_driver dsa_drv = { 9662306a36Sopenharmony_ci .name = "dsa", 9762306a36Sopenharmony_ci .probe = idxd_dsa_drv_probe, 9862306a36Sopenharmony_ci .remove = idxd_dsa_drv_remove, 9962306a36Sopenharmony_ci .type = dev_types, 10062306a36Sopenharmony_ci .drv = { 10162306a36Sopenharmony_ci .suppress_bind_attrs = true, 10262306a36Sopenharmony_ci .groups = dsa_drv_compat_groups, 10362306a36Sopenharmony_ci }, 10462306a36Sopenharmony_ci}; 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_cimodule_idxd_driver(dsa_drv); 10762306a36Sopenharmony_ciMODULE_IMPORT_NS(IDXD); 108