#include <linux/errno.h>
#include <linux/kernel.h>
#include <linux/major.h>
+#include <linux/kmod.h> /* for request_module() */
#include <linux/sched.h>
#include <linux/lp.h>
#include <linux/malloc.h>
#include <linux/obd_class.h>
static int obd_init_magic;
-int obd_print_entry = 1;
-int obd_debug_level = 4095;
+int obd_print_entry = 0;
+int obd_debug_level = D_IOCTL|D_INODE|D_SUPER|D_WARNING|D_MALLOC|D_CACHE;
+long obd_memory = 0;
struct obd_device obd_dev[MAX_OBD_DEVICES];
struct list_head obd_types;
return -ENODEV;
fsync_dev(inode->i_rdev);
if (obd_dev[dev].obd_refcnt <= 0)
- printk(KERN_ALERT "presto_psdev_release: refcount(%d) <= 0\n",
+ printk(KERN_ALERT __FUNCTION__ ": refcount(%d) <= 0\n",
obd_dev[dev].obd_refcnt);
obd_dev[dev].obd_refcnt--;
{
struct list_head *tmp;
struct obd_type *type;
- CDEBUG(D_IOCTL, "SEARCH %s\n", nm);
+ CDEBUG(D_INFO, "SEARCH %s\n", nm);
tmp = &obd_types;
while ( (tmp = tmp->next) != &obd_types ) {
type = list_entry(tmp, struct obd_type, typ_chain);
- CDEBUG(D_IOCTL, "TYP %s\n", type->typ_name);
+ CDEBUG(D_INFO, "TYP %s\n", type->typ_name);
if (strlen(type->typ_name) == strlen(nm) &&
strcmp(type->typ_name, nm) == 0 ) {
return type;
{
void *tmp = NULL;
- if (!len)
+ if (!len) {
+ *data = NULL;
return 0;
+ }
- CDEBUG(D_IOCTL, "getdata: len %d, add %p\n", len, *data);
+ CDEBUG(D_MALLOC, "len %d, add %p\n", len, *data);
OBD_ALLOC(tmp, void *, len);
if ( !tmp )
memset(tmp, 0, len);
if ( copy_from_user(tmp, *data, len)) {
- OBD_FREE(tmp,len);
+ OBD_FREE(tmp, len);
return -EFAULT;
}
*data = tmp;
int err, dev;
long int cli_id; /* connect, disconnect */
- if (!inode)
+ if (!inode) {
+ CDEBUG(D_IOCTL, "invalid inode\n");
return -EINVAL;
+ }
dev = MINOR(inode->i_rdev);
if (dev > MAX_OBD_DEVICES)
case OBD_IOC_ATTACH: {
struct obd_type *type;
struct oic_generic *input = tmp_buf;
+ char *nm;
ENTRY;
/* have we attached a type to this device */
}
/* find the type */
- type = obd_nm_to_type(input->att_type);
+ nm = input->att_type;
+ type = obd_nm_to_type(nm);
+#ifdef CONFIG_KMOD
+ if ( !type ) {
+ if ( !request_module(nm) ) {
+ CDEBUG(D_PSDEV, "Loaded module '%s'\n", nm);
+ type = obd_nm_to_type(nm);
+ } else {
+ CDEBUG(D_PSDEV, "Can't load module '%s'\n", nm);
+ }
+ }
+#endif
+
OBD_FREE(input->att_type, input->att_typelen + 1);
if ( !type ) {
- printk("Unknown obd type dev %d\n", dev);
+ printk(__FUNCTION__ ": unknown obd type dev %d\n",
+ dev);
EXIT;
return -EINVAL;
}
dev, input->att_datalen, obddev->obd_type->typ_name);
/* maybe we are done */
if ( !OBT(obddev) || !OBP(obddev, attach) ) {
- obddev->obd_flags |= OBD_ATTACHED;
+ obddev->obd_flags |= OBD_ATTACHED;
type->typ_refcnt++;
- CDEBUG(D_IOCTL, "Dev %d refcount now %d\n", dev,
+ CDEBUG(D_PSDEV, "Dev %d refcount now %d\n", dev,
type->typ_refcnt);
+ if (input->att_data)
+ OBD_FREE(input->att_data, input->att_datalen);
MOD_INC_USE_COUNT;
EXIT;
return 0;
/* do the attach */
err = OBP(obddev, attach)(obddev, input->att_datalen,
input->att_data);
- OBD_FREE(input->att_data, input->att_datalen);
+ if (input->att_data)
+ OBD_FREE(input->att_data, input->att_datalen);
if ( err ) {
obddev->obd_flags &= ~OBD_ATTACHED;
} else {
obddev->obd_flags |= OBD_ATTACHED;
type->typ_refcnt++;
- CDEBUG(D_IOCTL, "Dev %d refcount now %d\n", dev,
+ CDEBUG(D_PSDEV, "Dev %d refcount now %d\n", dev,
type->typ_refcnt);
MOD_INC_USE_COUNT;
EXIT;
return -EBUSY;
}
- CDEBUG(D_IOCTL, "Detach %d, type %s\n", dev,
+ CDEBUG(D_PSDEV, "Detach %d, type %s\n", dev,
obddev->obd_type->typ_name);
obddev->obd_flags &= ~OBD_ATTACHED;
obddev->obd_type->typ_refcnt--;
- CDEBUG(D_IOCTL, "Dev %d refcount now %d\n", dev,
+ CDEBUG(D_PSDEV, "Dev %d refcount now %d\n", dev,
obddev->obd_type->typ_refcnt);
obddev->obd_type = NULL;
MOD_DEC_USE_COUNT;
}
/* do the setup */
- CDEBUG(D_IOCTL, "Setup %d, type %s\n", dev,
+ CDEBUG(D_PSDEV, "Setup %d, type %s\n", dev,
obddev->obd_type->typ_name);
if ( !OBT(obddev) || !OBP(obddev, setup) ) {
obddev->obd_type->typ_refcnt++;
- CDEBUG(D_IOCTL, "Dev %d refcount now %d\n",
+ CDEBUG(D_PSDEV, "Dev %d refcount now %d\n",
dev, obddev->obd_type->typ_refcnt);
+ if (setup->setup_data)
+ OBD_FREE(setup->setup_data,
+ setup->setup_datalen);
obddev->obd_flags |= OBD_SET_UP;
EXIT;
return 0;
EXIT;
} else {
obddev->obd_type->typ_refcnt++;
- CDEBUG(D_IOCTL, "Dev %d refcount now %d\n",
+ CDEBUG(D_PSDEV, "Dev %d refcount now %d\n",
dev, obddev->obd_type->typ_refcnt);
obddev->obd_flags |= OBD_SET_UP;
EXIT;
}
+ if (setup->setup_data)
+ OBD_FREE(setup->setup_data, setup->setup_datalen);
return err;
}
case OBD_IOC_CLEANUP: {
}
if ( !obddev->obd_type->typ_refcnt ) {
- printk("OBD_CLEANUP: Dev %d has refcount (%d)!\n",
+ CDEBUG(D_IOCTL, "dev %d has refcount (%d)!\n",
dev, obddev->obd_type->typ_refcnt);
EXIT;
return -EBUSY;
if ( (!(obddev->obd_flags & OBD_SET_UP)) ||
(!(obddev->obd_flags & OBD_ATTACHED))) {
- CDEBUG(D_IOCTL, "Device not attached or set up\n");
+ CDEBUG(D_IOCTL, "device not attached or set up\n");
EXIT;
return -ENODEV;
}
cleanup_out:
obddev->obd_flags &= ~OBD_SET_UP;
obddev->obd_type->typ_refcnt--;
- CDEBUG(D_IOCTL, "Dev %d refcount now %d\n", dev,
+ CDEBUG(D_PSDEV, "Dev %d refcount now %d\n", dev,
obddev->obd_type->typ_refcnt);
EXIT;
return 0;
&rw_s->count, rw_s->offset);
ODEBUG(&rw_s->obdo);
- CDEBUG(D_INODE, "READ: conn %d, count %Ld, offset %Ld, '%s'\n",
+ CDEBUG(D_INFO, "READ: conn %d, count %Ld, offset %Ld, '%s'\n",
rw_s->conn_id, rw_s->count, rw_s->offset, rw_s->buf);
if ( err ) {
EXIT;
return err;
}
- CDEBUG(D_INODE, "WRITE: conn %d, count %Ld, offset %Ld, '%s'\n",
+ CDEBUG(D_INFO, "WRITE: conn %d, count %Ld, offset %Ld, '%s'\n",
rw_s->conn_id, rw_s->count, rw_s->offset, rw_s->buf);
err = OBP(obddev, write)(&conn, &rw_s->obdo, rw_s->buf,
&rw_s->count, rw_s->offset);
return -EOPNOTSUPP;
/* do the partition */
- CDEBUG(D_IOCTL, "Copy %d, type %s dst %Ld src %Ld\n", dev,
+ CDEBUG(D_INFO, "Copy %d, type %s dst %Ld src %Ld\n", dev,
obddev->obd_type->typ_name, mvdata->dst.o_id,
mvdata->src.o_id);
return err;
}
- CDEBUG(D_IOCTL, "Migrate copying %d bytes\n", sizeof(*mvdata));
+ CDEBUG(D_INFO, "Migrate copying %d bytes\n", sizeof(*mvdata));
if ( !OBT(obddev) || !OBP(obddev, migrate) )
return -EOPNOTSUPP;
/* do the partition */
- CDEBUG(D_IOCTL, "Migrate %d, type %s conn %d src %Ld dst %Ld\n",
+ CDEBUG(D_INFO, "Migrate %d, type %s conn %d src %Ld dst %Ld\n",
dev, obddev->obd_type->typ_name, mvdata->src_conn_id,
mvdata->src.o_id, mvdata->dst.o_id);
default: {
struct obd_type *type;
struct oic_generic input;
+ char *nm;
void *karg;
/* get data structures */
}
/* find the type */
- type = obd_nm_to_type(input.att_type);
+ nm = input.att_type;
+ type = obd_nm_to_type(nm);
+#ifdef CONFIG_KMOD
+ if ( !type ) {
+ if ( !request_module(nm) ) {
+ CDEBUG(D_PSDEV, "Loaded module '%s'\n", nm);
+ type = obd_nm_to_type(nm);
+ } else {
+ CDEBUG(D_PSDEV, "Can't load module '%s'\n", nm);
+ }
+ }
+#endif
OBD_FREE(input.att_type, input.att_typelen + 1);
if ( !type ) {
- printk("Unknown obd type dev %d\n", dev);
+ printk(__FUNCTION__ ": unknown obd type dev %d\n", dev);
EXIT;
return -EINVAL;
}
}
conn.oc_id = input.att_connid;
- CDEBUG(D_IOCTL, "Calling ioctl %x for type %s, len %d\n",
+ CDEBUG(D_INFO, "Calling ioctl %x for type %s, len %d\n",
cmd, type->typ_name, input.att_datalen);
/* get the generic data */
return err;
}
}
-}
+} /* obd_class_ioctl */
/* Driver interface done, utility functions follow */
if (obd_init_magic != 0x11223344) {
+ printk(__FUNCTION__ ": bad magic for type\n");
EXIT;
return -EINVAL;
}
if ( !type ) {
MOD_DEC_USE_COUNT;
- printk("Unknown obd type\n");
+ printk(KERN_INFO __FUNCTION__ ": unknown obd type\n");
EXIT;
return -EINVAL;
}
if ( type->typ_refcnt ) {
MOD_DEC_USE_COUNT;
- printk("OBD: Type %s has refcount (%d)\n", nm,
- type->typ_refcnt);
+ printk(KERN_ALERT __FUNCTION__ ":type %s has refcount "
+ "(%d)\n", nm, type->typ_refcnt);
EXIT;
return -EBUSY;
}
OBD_FREE(type, sizeof(*type));
MOD_DEC_USE_COUNT;
return 0;
-}
+} /* obd_unregister_type */
/* declare character device */
static struct file_operations obd_psdev_fops = {
obd_class_release, /* release */
NULL, /* fsync */
NULL, /* fasync */
- NULL, /* check_media_change */
- NULL, /* revalidate */
NULL /* lock */
};
if (register_chrdev(OBD_PSDEV_MAJOR,"obd_psdev",
&obd_psdev_fops)) {
- printk(KERN_ERR "obd_psdev: unable to get major %d\n",
+ printk(KERN_ERR __FUNCTION__ ": unable to get major %d\n",
OBD_PSDEV_MAJOR);
return -EIO;
}
obd_cleanup_obdo_cache();
obd_sysctl_clean();
+ CDEBUG(D_MALLOC, "CLASS mem used %ld\n", obd_memory);
obd_init_magic = 0;
EXIT;
}