/* * Interface for Dynamic Logical Partitioning of I/O Slots on * RPA-compliant PPC64 platform. * * John Rose <johnrose@austin.ibm.com> * October 2003 * * Copyright (C) 2003 IBM. * * 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. */ #include <linux/kobject.h> #include <linux/string.h> #include "pci_hotplug.h" #include "rpadlpar.h" #define DLPAR_KOBJ_NAME "control" #define ADD_SLOT_ATTR_NAME "add_slot" #define REMOVE_SLOT_ATTR_NAME "remove_slot" #define MAX_DRC_NAME_LEN 64 /* Store return code of dlpar operation in attribute struct */ struct dlpar_io_attr { int rc; struct attribute attr; ssize_t (*store)(struct dlpar_io_attr *dlpar_attr, const char *buf, size_t nbytes); }; /* Common show callback for all attrs, display the return code * of the dlpar op */ static ssize_t dlpar_attr_show(struct kobject * kobj, struct attribute * attr, char * buf) { struct dlpar_io_attr *dlpar_attr = container_of(attr, struct dlpar_io_attr, attr); return sprintf(buf, "%d\n", dlpar_attr->rc); } static ssize_t dlpar_attr_store(struct kobject * kobj, struct attribute * attr, const char *buf, size_t nbytes) { struct dlpar_io_attr *dlpar_attr = container_of(attr, struct dlpar_io_attr, attr); return dlpar_attr->store ? dlpar_attr->store(dlpar_attr, buf, nbytes) : -EIO; } static struct sysfs_ops dlpar_attr_sysfs_ops = { .show = dlpar_attr_show, .store = dlpar_attr_store, }; static ssize_t add_slot_store(struct dlpar_io_attr *dlpar_attr, const char *buf, size_t nbytes) { char drc_name[MAX_DRC_NAME_LEN]; char *end; if (nbytes >= MAX_DRC_NAME_LEN) return 0; memcpy(drc_name, buf, nbytes); end = strchr(drc_name, '\n'); if (!end) end = &drc_name[nbytes]; *end = '\0'; dlpar_attr->rc = dlpar_add_slot(drc_name); return nbytes; } static ssize_t remove_slot_store(struct dlpar_io_attr *dlpar_attr, const char *buf, size_t nbytes) { char drc_name[MAX_DRC_NAME_LEN]; char *end; if (nbytes >= MAX_DRC_NAME_LEN) return 0; memcpy(drc_name, buf, nbytes); end = strchr(drc_name, '\n'); if (!end) end = &drc_name[nbytes]; *end = '\0'; dlpar_attr->rc = dlpar_remove_slot(drc_name); return nbytes; } static struct dlpar_io_attr add_slot_attr = { .rc = 0, .attr = { .name = ADD_SLOT_ATTR_NAME, .mode = 0644, }, .store = add_slot_store, }; static struct dlpar_io_attr remove_slot_attr = { .rc = 0, .attr = { .name = REMOVE_SLOT_ATTR_NAME, .mode = 0644}, .store = remove_slot_store, }; static struct attribute *default_attrs[] = { &add_slot_attr.attr, &remove_slot_attr.attr, NULL, }; static void dlpar_io_release(struct kobject *kobj) { /* noop */ return; } struct kobj_type ktype_dlpar_io = { .release = dlpar_io_release, .sysfs_ops = &dlpar_attr_sysfs_ops, .default_attrs = default_attrs, }; struct kset dlpar_io_kset = { .subsys = &pci_hotplug_slots_subsys, .kobj = {.name = DLPAR_KOBJ_NAME, .ktype=&ktype_dlpar_io,}, .ktype = &ktype_dlpar_io, }; int dlpar_sysfs_init(void) { if (kset_register(&dlpar_io_kset)) { printk(KERN_ERR "rpadlpar_io: cannot register kset for %s\n", dlpar_io_kset.kobj.name); return -EINVAL; } return 0; } void dlpar_sysfs_exit(void) { kset_unregister(&dlpar_io_kset); }