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