1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright 2021 Xillybus Ltd, http://xillybus.com
5 * Driver for the Xillybus class
8 #include <linux/types.h>
9 #include <linux/module.h>
10 #include <linux/device.h>
12 #include <linux/cdev.h>
13 #include <linux/slab.h>
14 #include <linux/list.h>
15 #include <linux/mutex.h>
17 #include "xillybus_class.h"
19 MODULE_DESCRIPTION("Driver for Xillybus class");
20 MODULE_AUTHOR("Eli Billauer, Xillybus Ltd.");
21 MODULE_ALIAS("xillybus_class");
22 MODULE_LICENSE("GPL v2");
24 static DEFINE_MUTEX(unit_mutex);
25 static LIST_HEAD(unit_list);
26 static struct class *xillybus_class;
28 #define UNITNAMELEN 16
31 struct list_head list_entry;
35 char name[UNITNAMELEN];
41 int xillybus_init_chrdev(struct device *dev,
42 const struct file_operations *fops,
45 unsigned char *idt, unsigned int len,
47 const char *prefix, bool enumerate)
54 struct device *device;
56 struct xilly_unit *unit, *u;
58 unit = kzalloc(sizeof(*unit), GFP_KERNEL);
63 mutex_lock(&unit_mutex);
66 snprintf(unit->name, UNITNAMELEN, "%s", prefix);
68 for (i = 0; enumerate; i++) {
69 snprintf(unit->name, UNITNAMELEN, "%s_%02d",
73 list_for_each_entry(u, &unit_list, list_entry)
74 if (!strcmp(unit->name, u->name)) {
80 rc = alloc_chrdev_region(&mdev, 0, num_nodes, unit->name);
83 dev_warn(dev, "Failed to obtain major/minors");
87 unit->major = MAJOR(mdev);
88 unit->lowest_minor = MINOR(mdev);
89 unit->num_nodes = num_nodes;
90 unit->private_data = private_data;
92 unit->cdev = cdev_alloc();
95 goto unregister_chrdev;
97 unit->cdev->ops = fops;
98 unit->cdev->owner = owner;
100 rc = cdev_add(unit->cdev, MKDEV(unit->major, unit->lowest_minor),
103 dev_err(dev, "Failed to add cdev.\n");
104 /* kobject_put() is normally done by cdev_del() */
105 kobject_put(&unit->cdev->kobj);
106 goto unregister_chrdev;
109 for (i = 0; i < num_nodes; i++) {
110 namelen = strnlen(idt, len);
112 if (namelen == len) {
113 dev_err(dev, "IDT's list of names is too short. This is exceptionally weird, because its CRC is OK\n");
115 goto unroll_device_create;
118 snprintf(devname, sizeof(devname), "%s_%s",
124 device = device_create(xillybus_class,
127 i + unit->lowest_minor),
131 if (IS_ERR(device)) {
132 dev_err(dev, "Failed to create %s device. Aborting.\n",
135 goto unroll_device_create;
140 dev_err(dev, "IDT's list of names is too long. This is exceptionally weird, because its CRC is OK\n");
142 goto unroll_device_create;
145 list_add_tail(&unit->list_entry, &unit_list);
147 dev_info(dev, "Created %d device files.\n", num_nodes);
149 mutex_unlock(&unit_mutex);
153 unroll_device_create:
154 for (i--; i >= 0; i--)
155 device_destroy(xillybus_class, MKDEV(unit->major,
156 i + unit->lowest_minor));
158 cdev_del(unit->cdev);
161 unregister_chrdev_region(MKDEV(unit->major, unit->lowest_minor),
165 mutex_unlock(&unit_mutex);
171 EXPORT_SYMBOL(xillybus_init_chrdev);
173 void xillybus_cleanup_chrdev(void *private_data,
177 struct xilly_unit *unit = NULL, *iter;
179 mutex_lock(&unit_mutex);
181 list_for_each_entry(iter, &unit_list, list_entry)
182 if (iter->private_data == private_data) {
188 dev_err(dev, "Weird bug: Failed to find unit\n");
189 mutex_unlock(&unit_mutex);
193 for (minor = unit->lowest_minor;
194 minor < (unit->lowest_minor + unit->num_nodes);
196 device_destroy(xillybus_class, MKDEV(unit->major, minor));
198 cdev_del(unit->cdev);
200 unregister_chrdev_region(MKDEV(unit->major, unit->lowest_minor),
203 dev_info(dev, "Removed %d device files.\n",
206 list_del(&unit->list_entry);
209 mutex_unlock(&unit_mutex);
211 EXPORT_SYMBOL(xillybus_cleanup_chrdev);
213 int xillybus_find_inode(struct inode *inode,
214 void **private_data, int *index)
216 int minor = iminor(inode);
217 int major = imajor(inode);
218 struct xilly_unit *unit = NULL, *iter;
220 mutex_lock(&unit_mutex);
222 list_for_each_entry(iter, &unit_list, list_entry)
223 if (iter->major == major &&
224 minor >= iter->lowest_minor &&
225 minor < (iter->lowest_minor + iter->num_nodes)) {
231 mutex_unlock(&unit_mutex);
235 *private_data = unit->private_data;
236 *index = minor - unit->lowest_minor;
238 mutex_unlock(&unit_mutex);
241 EXPORT_SYMBOL(xillybus_find_inode);
243 static int __init xillybus_class_init(void)
245 xillybus_class = class_create(THIS_MODULE, "xillybus");
247 if (IS_ERR(xillybus_class)) {
248 pr_warn("Failed to register xillybus class\n");
250 return PTR_ERR(xillybus_class);
255 static void __exit xillybus_class_exit(void)
257 class_destroy(xillybus_class);
260 module_init(xillybus_class_init);
261 module_exit(xillybus_class_exit);