/* * * Copyright (c) International Business Machines Corp., 2001 * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See * the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ /* * This test module is for executing and testing * the kernel code from drivers/base. This module * is driven by a user space program through * calls to the ioctl * * author: Sean Ruyle * date: 07/14/2003 * * module: tbase */ #include <linux/types.h> #include <linux/kernel.h> #include <linux/fs.h> #include <linux/ioctl.h> #include <linux/module.h> #include <linux/init.h> #include <linux/device.h> #include <linux/pci.h> #include <linux/sysdev.h> #include <asm/uaccess.h> #include "tbase.h" #include "str_mod.h" MODULE_AUTHOR("Sean Ruyle <srruyle@us.ibm.com>"); MODULE_DESCRIPTION(TMOD_DRIVER_NAME); MODULE_LICENSE("GPL"); static int tbase_ioctl(struct inode *, struct file *, unsigned int, unsigned long); static int tbase_open(struct inode *, struct file *); static int tbase_close(struct inode *, struct file *); static int test_device_register(void); static int test_device_unregister(void); static int test_bus_add(void); static int test_get_drv(void); static int test_put_drv(void); static int test_reg_firm(void); static int test_create_file(void); static int test_dev_suspend(void); static int test_dev_file(void); static int test_bus_rescan(void); static int test_bus_file(void); static int test_class_reg(void); static int test_class_get(void); static int test_class_file(void); static int test_classdev_reg(void); static int test_classint_reg(void); static int test_sysdev_cls_reg(void); static int test_sysdev_reg(void); static int Major = TBASEMAJOR; static ltpmod_user_t ltp_mod; /* * File operations struct, to use operations find the * correct file descriptor */ static struct file_operations tbase_fops = { open: tbase_open, release:tbase_close, ioctl: tbase_ioctl, }; static int tbase_open(struct inode *ino, struct file *f) { return 0; } static int tbase_close(struct inode *ino, struct file *f) { return 0; } /* my bus stuff */ struct device_driver test_driver; struct device test_device; static int test_device_match(struct device *dev, struct device_driver *drv) { printk("tbase: driver is %s\n", drv->name); // printk("tbase: device is %s\n", dev->name); if (drv == &test_driver && dev == &test_device) { printk("tbase: match\n"); return 1; } else { printk("tbase: no match\n"); return 0; } } struct bus_type test_bus_type = { .name = "test_bus", .match = test_device_match, }; /* my driver stuff */ int test_dev_probe(struct device *dev) { printk("tbase: Entered test_dev_probe\n"); return 0; } int test_dev_remove(struct device *dev) { printk("tbase: Entered test_dev_remove\n"); return 0; } struct device_driver test_driver = { .name = "TestDriver", .bus = &test_bus_type, .probe = test_dev_probe, .remove = test_dev_remove, }; /* my device stuff */ struct device test_device = { // .name = "TestDevice", .bus = &test_bus_type, .bus_id = "test_bus", }; /* my class stuff */ static void test_class_release(struct class_device *class_dev) { printk("tbase: Entered test_class_release\n"); } int test_class_hotplug(struct class_device *dev, char **envp, int num_envp, char *buffer, int buffer_size) { printk("tbase: Entered test_class_hotplug\n"); return 0; } struct class test_class = { .name = "TestClass", .hotplug = test_class_hotplug, .release = test_class_release, }; /* my class device stuff */ struct class_device test_class_dev = { .class_id = "test_bus", .dev = &test_device, .class = &test_class, }; /* my class interface stuff */ int test_intf_add(struct class_device *class_dev) { printk("tbase: Entered test_intf_add for the test class_interface\n"); return 0; } void test_intf_rem(struct class_device *class_dev) { printk("tbase: Entered test_intf_rem for the test class interface\n"); } struct class_interface test_interface = { .class = &test_class, .add = &test_intf_add, .remove = &test_intf_rem, }; /* my sys_device stuff */ int test_resume(struct sys_device *dev) { printk("tbase: Entered test resume for sys device\n"); return 0; } struct sysdev_class test_sysclass = { set_kset_name("TestSysclass"), .resume = test_resume, }; struct sys_device test_sys_device = { .id = 0, .cls = &test_sysclass, }; /* my attribute stuff */ static inline ssize_t store_new_id(struct device_driver *driver, const char *buf, size_t count) { printk("tbase: Entered store new id\n"); return count; } /* create attribute driver_attr_new_id */ DRIVER_ATTR(new_id, 0200, NULL, store_new_id); /* create attribute dev_attr_test_id */ DEVICE_ATTR(test_id, S_IRUGO, NULL, NULL); /* create attribute bus_attr_test_id */ BUS_ATTR(test_id, S_IRUGO, NULL, NULL); /* create attribute class_attr_test_id */ CLASS_ATTR(test_id, 0644, NULL, NULL); /* create attribute class_device_attr_test_id */ CLASS_DEVICE_ATTR(test_id, 0644, NULL, NULL); /* * tbase_ioctl: * a user space program can drive the test functions * through a call to ioctl once the correct file * descriptor has been attained */ static int tbase_ioctl(struct inode *ino, struct file *f, unsigned int cmd, unsigned long l) { int rc; tmod_interface_t tif; caddr_t *inparms; caddr_t *outparms; printk("Enter tbase_ioctl\n"); inparms = NULL; outparms = NULL; rc = 0; /* * the following calls are used to setup the * parameters that might need to be passed * between user and kernel space, using the tif * pointer that is passed in as the last * parameter to the ioctl * */ if (copy_from_user(&tif, (void *)l, sizeof(tif))) { /* Bad address */ return (-EFAULT); } /* * Setup inparms and outparms as needed */ if (tif.in_len > 0) { inparms = (caddr_t *) kmalloc(tif.in_len, GFP_KERNEL); if (!inparms) { return (-ENOMEM); } rc = copy_from_user(inparms, tif.in_data, tif.in_len); if (rc) { kfree(inparms); return (-EFAULT); } } if (tif.out_len > 0) { outparms = (caddr_t *) kmalloc(tif.out_len, GFP_KERNEL); if (!outparms) { kfree(inparms); return (-ENOMEM); } } /* * Use a switch statement to determine which function * to call, based on the cmd flag that is specified * in user space. Pass in inparms or outparms as * needed * */ switch (cmd) { case REG_DEVICE: rc = test_device_register(); break; case UNREG_DEVICE: rc = test_device_unregister(); break; case BUS_ADD: rc = test_bus_add(); break; case GET_DRV: rc = test_get_drv(); break; case PUT_DRV: rc = test_put_drv(); break; case REG_FIRM: rc = test_reg_firm(); break; case CREATE_FILE: rc = test_create_file(); break; case DEV_SUSPEND: rc = test_dev_suspend(); break; case DEV_FILE: rc = test_dev_file(); break; case BUS_RESCAN: rc = test_bus_rescan(); break; case BUS_FILE: rc = test_bus_file(); break; case CLASS_REG: rc = test_class_reg(); break; case CLASS_UNREG: class_unregister(&test_class); break; case CLASS_GET: rc = test_class_get(); break; case CLASS_FILE: rc = test_class_file(); break; case CLASSDEV_REG: rc = test_classdev_reg(); break; case CLASSINT_REG: rc = test_classint_reg(); break; case SYSDEV_CLS_REG: rc = test_sysdev_cls_reg(); break; case SYSDEV_CLS_UNREG: sysdev_class_unregister(&test_sysclass); break; case SYSDEV_REG: rc = test_sysdev_reg(); break; case SYSDEV_UNREG: sys_device_unregister(&test_sys_device); break; default: printk("tbase: Mismatching ioctl command\n"); break; } /* * copy in the test return code, the reason we * this is so that in user space we can tell the * difference between an error in one of our test * calls or an error in the ioctl function */ tif.out_rc = rc; rc = 0; /* * setup the rest of tif pointer for returning to * to user space, using copy_to_user if needed */ /* if outparms then copy outparms into tif.out_data */ if (outparms) { if (copy_to_user(tif.out_data, outparms, tif.out_len)) { printk ("tbase: Unsuccessful copy_to_user of outparms\n"); rc = -EFAULT; } } /* copy tif structure into l so that can be used by user program */ if (copy_to_user((void *)l, &tif, sizeof(tif))) { printk("tbase: Unsuccessful copy_to_user of tif\n"); rc = -EFAULT; } /* * free inparms and outparms */ if (inparms) { kfree(inparms); } if (outparms) { kfree(outparms); } return rc; } /* * test_device_register * makes call to device register passing in * the device pointer that we found in a previos * function, returns an error code */ static int test_device_register() { struct device *dev = ltp_mod.dev; struct device_driver *drv = dev->driver; /* check if device register returns an error */ if (device_register(dev)) { printk("tbase: Device not registered\n"); return 1; } else printk("tbase: Device registered\n"); driver_unregister(drv); /* check if driver_register returns an error */ if (driver_register(drv)) { printk("tbase: Driver not registered\n"); return 1; } else printk("tbase: Driver registered\n"); return 0; } /* * test_device_unregister * make test call to device_unregister which * will in turn make calls that will decrememnt * the reference count and clean up as required */ static int test_device_unregister() { struct device *dev = ltp_mod.dev; /* increment reference count */ get_device(dev); /* reset remove pointer */ if (dev->driver->remove) dev->driver->remove = NULL; device_unregister(dev); //check that reference count is smaller by one return 0; } /* * test_bus_add * make call to bus_add_device, which will * in turn add the device that is passed in * to the bus */ static int test_bus_add() { /* check if device register returns an error */ if (bus_add_device(&test_device)) { printk("tbase: Device not added to bus\n"); return 1; } else { printk("tbase: Device added to bus\n"); return 0; } } /* * test_get_drv * make test call to get_driver which should * return a pointer to the driver passed in * and increase the reference count to that * kobject */ static int test_get_drv() { int a, rc; struct device_driver *drv = &test_driver, *tmp = NULL; /* get reference count before test call */ a = atomic_read(&drv->kobj.refcount); /* make test call */ if ((tmp = get_driver(drv))) { rc = 0; printk("tbase: get driver returned driver\n"); } else { rc = 1; printk("tbase: get driver failed to return driver\n"); } /* check reference count */ if ((a == (atomic_read(&drv->kobj.refcount) - 1))) { rc = 0; printk("tbase: correctly set ref count get driver\n"); } else { rc = 1; printk("tbase: incorrect ref count get driver\n"); } return rc; } /* * test_class_get * make test call to class_get which should return * a pointer to the class passed in and increase * the reference count to that kobject */ static int test_class_get() { int rc; struct class *tmp = NULL; /* get reference count before test call */ tmp = class_get(&test_class); if (tmp == &test_class) { printk("tbase: Success get class\n"); rc = 0; } else { printk("tbase: Failure get class\n"); rc = 1; } class_put(&test_class); return rc; } /* * test_put_drv * make test call to put_driver which should * decrease the reference count to the kobject * pointer in the driver structure */ static int test_put_drv() { int a, rc; struct device_driver *drv = &test_driver; /* get reference count before test call */ a = atomic_read(&drv->kobj.refcount); /* make test call */ put_driver(drv); /* check reference count */ if ((a == (atomic_read(&drv->kobj.refcount) + 1))) { rc = 0; printk("tbase: correctly set ref count put driver\n"); } else { rc = 1; printk("tbase: incorrect ref count put driver\n"); } return rc; } /* * test_reg_firm * test call to register_firmware, which will * register the subsystem, takes in a struct * subsystem pointer, we can use our bus pointer * that should have been found in a previous test * to pass in a subsystem pointer, returns an * error code */ static int test_reg_firm() { struct subsystem *subsys = NULL; /* check pointer exists */ if (!(subsys = &test_bus_type.subsys)) { printk("tbase: subsys pointer not set in reg firmware\n"); return 1; } /* unregiser firmware */ firmware_unregister(subsys); /* make test call */ if (firmware_register(subsys)) { printk("tbase: failed register firmware\n"); return 1; } else { printk("tbase: regsitered firmware\n"); return 0; } } /* * test_create_file * make test call to create sysfs file for the * driver and if that call is successful then * make a call to remove the file */ static int test_create_file() { struct device_driver *drv = &test_driver; if (driver_create_file(drv, &driver_attr_new_id)) { printk("tbase: failed create sysfs file\n"); return 1; } else { printk("tbase: created sysfs file\n"); driver_remove_file(drv, &driver_attr_new_id); return 0; } } /* * test_dev_suspend * make test call to device_suspend and * if that call is successful then make * a call to device_resume */ static int test_dev_suspend() { int error = 0; error = device_suspend(SUSPEND_SAVE_STATE); if (error) printk("tbase: Failed on device suspend call\n"); else { printk("tbase: Successful on device suspend call\n"); device_resume(); } error = device_suspend(SUSPEND_DISABLE); if (error) printk("tbase: Failed on device suspend call\n"); else { printk("tbase: Successful on device suspend call\n"); device_resume(); } return error; } /* * test_dev_file * make test call to device_create_file * and if that call is successful make * another call to device_remove_file */ static int test_dev_file() { struct device *dev = &test_device; if (device_create_file(dev, &dev_attr_test_id)) { printk("tbase: failed to create dev sysfs file\n"); return 1; } else { printk("tbase: created dev sysfs file\n"); device_remove_file(dev, &dev_attr_test_id); return 0; } } /* * test_bus_rescan * make test call to bus_rescan_devices which * will rescan the bus and attempt to match devices * to drivers, will return 0 for no matches or * the number of matches made, check that the * value returned is not negative */ static int test_bus_rescan() { int count = 0; count = bus_rescan_devices(&test_bus_type); if (count == 0) printk("tbase: found no device/driver matches\n"); else if (count > 0) printk("tbase; found match\n"); else { printk("tbase: bus rescan failed\n"); return count; } return 0; } /* * test_bus_file * make test call to bus_create_file * and if that call is successful make * another call to bus_remove_file */ static int test_bus_file() { struct bus_type *bus = &test_bus_type; if (bus_create_file(bus, &bus_attr_test_id)) { printk("tbase: failed to create bus sysfs file\n"); return 1; } else { printk("tbase: created bus sysfs file\n"); bus_remove_file(bus, &bus_attr_test_id); return 0; } } /* * test_class_file * make test call to class_create_file * and if that call is successful make * another call to class_remove_file */ static int test_class_file() { struct class *cls = &test_class; if (class_create_file(cls, &class_attr_test_id)) { printk("tbase: failed to create class sysfs file\n"); return 1; } else { printk("tbase: created class sysfs file\n"); class_remove_file(cls, &class_attr_test_id); return 0; } } /* * test_class_reg * make test call to class_register * with the test_class that is defined * in this module, if that call is * successful then call unregister */ static int test_class_reg() { int error; error = class_register(&test_class); if (error) printk("tbase: class register failed\n"); else printk("tbase: class register succeeded\n"); return error; } /* * test_classdev_reg * make test call to class_device_register * and if that returns successful then * make call to class_device_unregister */ static int test_classdev_reg() { int rc = 0; if (class_device_register(&test_class_dev)) { printk("tbase: Failed to register class device\n"); rc = 1; } else { printk("tbase: Registered class device\n"); /* make class device sysfs file */ if (class_device_create_file (&test_class_dev, &class_device_attr_test_id)) { rc = 1; printk ("tbase: Failed to create class device sysfs file\n"); } else { printk("tbase: Created class device sysfs file\n"); class_device_remove_file(&test_class_dev, &class_device_attr_test_id); } class_device_unregister(&test_class_dev); } return rc; } /* * test_classint_reg * make test call to class_interface_register * and if that returns successfule then * make call to class_interface_unregister */ static int test_classint_reg() { if (class_interface_register(&test_interface)) { printk("tbase: Failed to register class interface\n"); return 1; } else { printk("tbase: Registered class interface\n"); class_interface_unregister(&test_interface); return 0; } } /* * test_sysdev_cls_reg * make test call to sysdev_class_register * to register the test_sysclass pointer * as a sysdev_class with the system, check * the return code */ static int test_sysdev_cls_reg() { if (sysdev_class_register(&test_sysclass)) { printk("tbase: Failed to register sysdev class\n"); return 1; } else { printk("tbase: Registered sysdev class\n"); return 0; } } /* * test_sysdev_reg * make test call to sys_device_register * to register the test_sysdev pointer * as a sys_device with the system, check * the return code */ static int test_sysdev_reg() { if (sys_device_register(&test_sys_device)) { printk("tbase: Failed to register sysdev \n"); return 1; } else { printk("tbase: Registered sysdev \n"); return 0; } } /* * tbase_init_module * set the owner of tbase_fops, register the module * as a char device, and perform any necessary * initialization */ static int tbase_init_module(void) { int rc; bus_register(&test_bus_type); driver_register(&test_driver); device_register(&test_device); tbase_fops.owner = THIS_MODULE; printk("tbase: *** Register device %s **\n", DEVICE_NAME); rc = register_chrdev(Major, DEVICE_NAME, &tbase_fops); if (rc < 0) { printk("tbase: Failed to register device.\n"); return rc; } if (Major == 0) Major = rc; /* call any other init functions you might use here */ printk("tbase: Registration success.\n"); return 0; } /* * tmod_exit_module * unregister the device and any necessary * operations to close devices */ static void tbase_exit_module(void) { int rc; device_unregister(&test_device); driver_unregister(&test_driver); bus_unregister(&test_bus_type); /* free any pointers still allocated, using kfree */ rc = unregister_chrdev(Major, DEVICE_NAME); if (rc < 0) printk("tbase: unregister failed\n"); else printk("tbase: unregister success\n"); } /* specify what that init is run when the module is first loaded and that exit is run when it is removed */ module_init(tbase_init_module) module_exit(tbase_exit_module)