Path: blob/master/drivers/char/xillybus/xillybus_class.c
26282 views
// SPDX-License-Identifier: GPL-2.0-only1/*2* Copyright 2021 Xillybus Ltd, http://xillybus.com3*4* Driver for the Xillybus class5*/67#include <linux/types.h>8#include <linux/module.h>9#include <linux/device.h>10#include <linux/fs.h>11#include <linux/cdev.h>12#include <linux/slab.h>13#include <linux/list.h>14#include <linux/mutex.h>1516#include "xillybus_class.h"1718MODULE_DESCRIPTION("Driver for Xillybus class");19MODULE_AUTHOR("Eli Billauer, Xillybus Ltd.");20MODULE_ALIAS("xillybus_class");21MODULE_LICENSE("GPL v2");2223static DEFINE_MUTEX(unit_mutex);24static LIST_HEAD(unit_list);25static const struct class xillybus_class = {26.name = "xillybus",27};2829#define UNITNAMELEN 163031struct xilly_unit {32struct list_head list_entry;33void *private_data;3435struct cdev *cdev;36char name[UNITNAMELEN];37int major;38int lowest_minor;39int num_nodes;40};4142int xillybus_init_chrdev(struct device *dev,43const struct file_operations *fops,44struct module *owner,45void *private_data,46unsigned char *idt, unsigned int len,47int num_nodes,48const char *prefix, bool enumerate)49{50int rc;51dev_t mdev;52int i;53char devname[48];5455struct device *device;56size_t namelen;57struct xilly_unit *unit, *u;5859unit = kzalloc(sizeof(*unit), GFP_KERNEL);6061if (!unit)62return -ENOMEM;6364mutex_lock(&unit_mutex);6566if (!enumerate)67snprintf(unit->name, UNITNAMELEN, "%s", prefix);6869for (i = 0; enumerate; i++) {70snprintf(unit->name, UNITNAMELEN, "%s_%02d",71prefix, i);7273enumerate = false;74list_for_each_entry(u, &unit_list, list_entry)75if (!strcmp(unit->name, u->name)) {76enumerate = true;77break;78}79}8081rc = alloc_chrdev_region(&mdev, 0, num_nodes, unit->name);8283if (rc) {84dev_warn(dev, "Failed to obtain major/minors");85goto fail_obtain;86}8788unit->major = MAJOR(mdev);89unit->lowest_minor = MINOR(mdev);90unit->num_nodes = num_nodes;91unit->private_data = private_data;9293unit->cdev = cdev_alloc();94if (!unit->cdev) {95rc = -ENOMEM;96goto unregister_chrdev;97}98unit->cdev->ops = fops;99unit->cdev->owner = owner;100101rc = cdev_add(unit->cdev, MKDEV(unit->major, unit->lowest_minor),102unit->num_nodes);103if (rc) {104dev_err(dev, "Failed to add cdev.\n");105/* kobject_put() is normally done by cdev_del() */106kobject_put(&unit->cdev->kobj);107goto unregister_chrdev;108}109110for (i = 0; i < num_nodes; i++) {111namelen = strnlen(idt, len);112113if (namelen == len) {114dev_err(dev, "IDT's list of names is too short. This is exceptionally weird, because its CRC is OK\n");115rc = -ENODEV;116goto unroll_device_create;117}118119snprintf(devname, sizeof(devname), "%s_%s",120unit->name, idt);121122len -= namelen + 1;123idt += namelen + 1;124125device = device_create(&xillybus_class,126NULL,127MKDEV(unit->major,128i + unit->lowest_minor),129NULL,130"%s", devname);131132if (IS_ERR(device)) {133dev_err(dev, "Failed to create %s device. Aborting.\n",134devname);135rc = -ENODEV;136goto unroll_device_create;137}138}139140if (len) {141dev_err(dev, "IDT's list of names is too long. This is exceptionally weird, because its CRC is OK\n");142rc = -ENODEV;143goto unroll_device_create;144}145146list_add_tail(&unit->list_entry, &unit_list);147148dev_info(dev, "Created %d device files.\n", num_nodes);149150mutex_unlock(&unit_mutex);151152return 0;153154unroll_device_create:155for (i--; i >= 0; i--)156device_destroy(&xillybus_class, MKDEV(unit->major,157i + unit->lowest_minor));158159cdev_del(unit->cdev);160161unregister_chrdev:162unregister_chrdev_region(MKDEV(unit->major, unit->lowest_minor),163unit->num_nodes);164165fail_obtain:166mutex_unlock(&unit_mutex);167168kfree(unit);169170return rc;171}172EXPORT_SYMBOL(xillybus_init_chrdev);173174void xillybus_cleanup_chrdev(void *private_data,175struct device *dev)176{177int minor;178struct xilly_unit *unit = NULL, *iter;179180mutex_lock(&unit_mutex);181182list_for_each_entry(iter, &unit_list, list_entry)183if (iter->private_data == private_data) {184unit = iter;185break;186}187188if (!unit) {189dev_err(dev, "Weird bug: Failed to find unit\n");190mutex_unlock(&unit_mutex);191return;192}193194for (minor = unit->lowest_minor;195minor < (unit->lowest_minor + unit->num_nodes);196minor++)197device_destroy(&xillybus_class, MKDEV(unit->major, minor));198199cdev_del(unit->cdev);200201unregister_chrdev_region(MKDEV(unit->major, unit->lowest_minor),202unit->num_nodes);203204dev_info(dev, "Removed %d device files.\n",205unit->num_nodes);206207list_del(&unit->list_entry);208kfree(unit);209210mutex_unlock(&unit_mutex);211}212EXPORT_SYMBOL(xillybus_cleanup_chrdev);213214int xillybus_find_inode(struct inode *inode,215void **private_data, int *index)216{217int minor = iminor(inode);218int major = imajor(inode);219struct xilly_unit *unit = NULL, *iter;220221mutex_lock(&unit_mutex);222223list_for_each_entry(iter, &unit_list, list_entry)224if (iter->major == major &&225minor >= iter->lowest_minor &&226minor < (iter->lowest_minor + iter->num_nodes)) {227unit = iter;228break;229}230231if (!unit) {232mutex_unlock(&unit_mutex);233return -ENODEV;234}235236*private_data = unit->private_data;237*index = minor - unit->lowest_minor;238239mutex_unlock(&unit_mutex);240return 0;241}242EXPORT_SYMBOL(xillybus_find_inode);243244static int __init xillybus_class_init(void)245{246return class_register(&xillybus_class);247}248249static void __exit xillybus_class_exit(void)250{251class_unregister(&xillybus_class);252}253254module_init(xillybus_class_init);255module_exit(xillybus_class_exit);256257258