xref: /openbmc/linux/drivers/dma/idxd/compat.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
16e7f3ee9SDave Jiang // SPDX-License-Identifier: GPL-2.0
26e7f3ee9SDave Jiang /* Copyright(c) 2021 Intel Corporation. All rights rsvd. */
36e7f3ee9SDave Jiang #include <linux/init.h>
46e7f3ee9SDave Jiang #include <linux/kernel.h>
56e7f3ee9SDave Jiang #include <linux/module.h>
66e7f3ee9SDave Jiang #include <linux/device.h>
76e7f3ee9SDave Jiang #include <linux/device/bus.h>
86e7f3ee9SDave Jiang #include "idxd.h"
96e7f3ee9SDave Jiang 
106e7f3ee9SDave Jiang extern int device_driver_attach(struct device_driver *drv, struct device *dev);
116e7f3ee9SDave Jiang extern void device_driver_detach(struct device *dev);
126e7f3ee9SDave Jiang 
136e7f3ee9SDave Jiang #define DRIVER_ATTR_IGNORE_LOCKDEP(_name, _mode, _show, _store)	\
146e7f3ee9SDave Jiang 	struct driver_attribute driver_attr_##_name =		\
156e7f3ee9SDave Jiang 	__ATTR_IGNORE_LOCKDEP(_name, _mode, _show, _store)
166e7f3ee9SDave Jiang 
unbind_store(struct device_driver * drv,const char * buf,size_t count)176e7f3ee9SDave Jiang static ssize_t unbind_store(struct device_driver *drv, const char *buf, size_t count)
186e7f3ee9SDave Jiang {
19*790f3b60SGreg Kroah-Hartman 	const struct bus_type *bus = drv->bus;
206e7f3ee9SDave Jiang 	struct device *dev;
216e7f3ee9SDave Jiang 	int rc = -ENODEV;
226e7f3ee9SDave Jiang 
236e7f3ee9SDave Jiang 	dev = bus_find_device_by_name(bus, NULL, buf);
246e7f3ee9SDave Jiang 	if (dev && dev->driver) {
256e7f3ee9SDave Jiang 		device_driver_detach(dev);
266e7f3ee9SDave Jiang 		rc = count;
276e7f3ee9SDave Jiang 	}
286e7f3ee9SDave Jiang 
296e7f3ee9SDave Jiang 	return rc;
306e7f3ee9SDave Jiang }
316e7f3ee9SDave Jiang static DRIVER_ATTR_IGNORE_LOCKDEP(unbind, 0200, NULL, unbind_store);
326e7f3ee9SDave Jiang 
bind_store(struct device_driver * drv,const char * buf,size_t count)336e7f3ee9SDave Jiang static ssize_t bind_store(struct device_driver *drv, const char *buf, size_t count)
346e7f3ee9SDave Jiang {
35*790f3b60SGreg Kroah-Hartman 	const struct bus_type *bus = drv->bus;
366e7f3ee9SDave Jiang 	struct device *dev;
37568b2126SDave Jiang 	struct device_driver *alt_drv = NULL;
386e7f3ee9SDave Jiang 	int rc = -ENODEV;
396e7f3ee9SDave Jiang 	struct idxd_dev *idxd_dev;
406e7f3ee9SDave Jiang 
416e7f3ee9SDave Jiang 	dev = bus_find_device_by_name(bus, NULL, buf);
426e7f3ee9SDave Jiang 	if (!dev || dev->driver || drv != &dsa_drv.drv)
436e7f3ee9SDave Jiang 		return -ENODEV;
446e7f3ee9SDave Jiang 
456e7f3ee9SDave Jiang 	idxd_dev = confdev_to_idxd_dev(dev);
466e7f3ee9SDave Jiang 	if (is_idxd_dev(idxd_dev)) {
476e7f3ee9SDave Jiang 		alt_drv = driver_find("idxd", bus);
486e7f3ee9SDave Jiang 	} else if (is_idxd_wq_dev(idxd_dev)) {
496e7f3ee9SDave Jiang 		struct idxd_wq *wq = confdev_to_wq(dev);
506e7f3ee9SDave Jiang 
51e9c5b0b5SChristophe JAILLET 		if (is_idxd_wq_kernel(wq))
526e7f3ee9SDave Jiang 			alt_drv = driver_find("dmaengine", bus);
53e9c5b0b5SChristophe JAILLET 		else if (is_idxd_wq_user(wq))
546e7f3ee9SDave Jiang 			alt_drv = driver_find("user", bus);
55e9c5b0b5SChristophe JAILLET 	}
566e7f3ee9SDave Jiang 	if (!alt_drv)
576e7f3ee9SDave Jiang 		return -ENODEV;
586e7f3ee9SDave Jiang 
596e7f3ee9SDave Jiang 	rc = device_driver_attach(alt_drv, dev);
606e7f3ee9SDave Jiang 	if (rc < 0)
616e7f3ee9SDave Jiang 		return rc;
626e7f3ee9SDave Jiang 
636e7f3ee9SDave Jiang 	return count;
646e7f3ee9SDave Jiang }
656e7f3ee9SDave Jiang static DRIVER_ATTR_IGNORE_LOCKDEP(bind, 0200, NULL, bind_store);
666e7f3ee9SDave Jiang 
676e7f3ee9SDave Jiang static struct attribute *dsa_drv_compat_attrs[] = {
686e7f3ee9SDave Jiang 	&driver_attr_bind.attr,
696e7f3ee9SDave Jiang 	&driver_attr_unbind.attr,
706e7f3ee9SDave Jiang 	NULL,
716e7f3ee9SDave Jiang };
726e7f3ee9SDave Jiang 
736e7f3ee9SDave Jiang static const struct attribute_group dsa_drv_compat_attr_group = {
746e7f3ee9SDave Jiang 	.attrs = dsa_drv_compat_attrs,
756e7f3ee9SDave Jiang };
766e7f3ee9SDave Jiang 
776e7f3ee9SDave Jiang static const struct attribute_group *dsa_drv_compat_groups[] = {
786e7f3ee9SDave Jiang 	&dsa_drv_compat_attr_group,
796e7f3ee9SDave Jiang 	NULL,
806e7f3ee9SDave Jiang };
816e7f3ee9SDave Jiang 
idxd_dsa_drv_probe(struct idxd_dev * idxd_dev)826e7f3ee9SDave Jiang static int idxd_dsa_drv_probe(struct idxd_dev *idxd_dev)
836e7f3ee9SDave Jiang {
846e7f3ee9SDave Jiang 	return -ENODEV;
856e7f3ee9SDave Jiang }
866e7f3ee9SDave Jiang 
idxd_dsa_drv_remove(struct idxd_dev * idxd_dev)876e7f3ee9SDave Jiang static void idxd_dsa_drv_remove(struct idxd_dev *idxd_dev)
886e7f3ee9SDave Jiang {
896e7f3ee9SDave Jiang }
906e7f3ee9SDave Jiang 
916e7f3ee9SDave Jiang static enum idxd_dev_type dev_types[] = {
926e7f3ee9SDave Jiang 	IDXD_DEV_NONE,
936e7f3ee9SDave Jiang };
946e7f3ee9SDave Jiang 
956e7f3ee9SDave Jiang struct idxd_device_driver dsa_drv = {
966e7f3ee9SDave Jiang 	.name = "dsa",
976e7f3ee9SDave Jiang 	.probe = idxd_dsa_drv_probe,
986e7f3ee9SDave Jiang 	.remove = idxd_dsa_drv_remove,
996e7f3ee9SDave Jiang 	.type = dev_types,
1006e7f3ee9SDave Jiang 	.drv = {
1016e7f3ee9SDave Jiang 		.suppress_bind_attrs = true,
1026e7f3ee9SDave Jiang 		.groups = dsa_drv_compat_groups,
1036e7f3ee9SDave Jiang 	},
1046e7f3ee9SDave Jiang };
1056e7f3ee9SDave Jiang 
1066e7f3ee9SDave Jiang module_idxd_driver(dsa_drv);
1076e7f3ee9SDave Jiang MODULE_IMPORT_NS(IDXD);
108