Commit da62aa69 authored by Linus Torvalds's avatar Linus Torvalds

Merge branch 'linux_next' of git://git.kernel.org/pub/scm/linux/kernel/git/mchehab/i7core

* 'linux_next' of git://git.kernel.org/pub/scm/linux/kernel/git/mchehab/i7core: (34 commits)
  i7core_edac: return -ENODEV when devices were already probed
  i7core_edac: properly terminate pci_dev_table
  i7core_edac: Avoid PCI refcount to reach zero on successive load/reload
  i7core_edac: Fix refcount error at PCI devices
  i7core_edac: it is safe to i7core_unregister_mci() when mci=NULL
  i7core_edac: Fix an oops at i7core probe
  i7core_edac: Remove unused member channels in i7core_pvt
  i7core_edac: Remove unused arg csrow from get_dimm_config
  i7core_edac: Reduce args of i7core_register_mci
  i7core_edac: Introduce i7core_unregister_mci
  i7core_edac: Use saved pointers
  i7core_edac: Check probe counter in i7core_remove
  i7core_edac: Call pci_dev_put() when alloc_i7core_dev()  failed
  i7core_edac: Fix error path of i7core_register_mci
  i7core_edac: Fix order of lines in i7core_register_mci
  i7core_edac: Always do get/put for all devices
  i7core_edac: Introduce i7core_pci_ctl_create/release
  i7core_edac: Introduce free_i7core_dev
  i7core_edac: Introduce alloc_i7core_dev
  i7core_edac: Reduce args of i7core_get_onedevice
  ...
parents f1ebdd60 76a7bd81
...@@ -42,8 +42,10 @@ ...@@ -42,8 +42,10 @@
#if PAGE_SHIFT < 20 #if PAGE_SHIFT < 20
#define PAGES_TO_MiB( pages ) ( ( pages ) >> ( 20 - PAGE_SHIFT ) ) #define PAGES_TO_MiB( pages ) ( ( pages ) >> ( 20 - PAGE_SHIFT ) )
#define MiB_TO_PAGES(mb) ((mb) >> (20 - PAGE_SHIFT))
#else /* PAGE_SHIFT > 20 */ #else /* PAGE_SHIFT > 20 */
#define PAGES_TO_MiB( pages ) ( ( pages ) << ( PAGE_SHIFT - 20 ) ) #define PAGES_TO_MiB( pages ) ( ( pages ) << ( PAGE_SHIFT - 20 ) )
#define MiB_TO_PAGES(mb) ((mb) >> (PAGE_SHIFT - 20))
#endif #endif
#define edac_printk(level, prefix, fmt, arg...) \ #define edac_printk(level, prefix, fmt, arg...) \
...@@ -328,7 +330,7 @@ struct csrow_info { ...@@ -328,7 +330,7 @@ struct csrow_info {
struct mcidev_sysfs_group { struct mcidev_sysfs_group {
const char *name; /* group name */ const char *name; /* group name */
struct mcidev_sysfs_attribute *mcidev_attr; /* group attributes */ const struct mcidev_sysfs_attribute *mcidev_attr; /* group attributes */
}; };
struct mcidev_sysfs_group_kobj { struct mcidev_sysfs_group_kobj {
...@@ -336,7 +338,7 @@ struct mcidev_sysfs_group_kobj { ...@@ -336,7 +338,7 @@ struct mcidev_sysfs_group_kobj {
struct kobject kobj; /* kobj for the group */ struct kobject kobj; /* kobj for the group */
struct mcidev_sysfs_group *grp; /* group description table */ const struct mcidev_sysfs_group *grp; /* group description table */
struct mem_ctl_info *mci; /* the parent */ struct mem_ctl_info *mci; /* the parent */
}; };
...@@ -347,7 +349,7 @@ struct mcidev_sysfs_group_kobj { ...@@ -347,7 +349,7 @@ struct mcidev_sysfs_group_kobj {
struct mcidev_sysfs_attribute { struct mcidev_sysfs_attribute {
/* It should use either attr or grp */ /* It should use either attr or grp */
struct attribute attr; struct attribute attr;
struct mcidev_sysfs_group *grp; /* Points to a group of attributes */ const struct mcidev_sysfs_group *grp; /* Points to a group of attributes */
/* Ops for show/store values at the attribute - not used on group */ /* Ops for show/store values at the attribute - not used on group */
ssize_t (*show)(struct mem_ctl_info *,char *); ssize_t (*show)(struct mem_ctl_info *,char *);
...@@ -440,7 +442,7 @@ struct mem_ctl_info { ...@@ -440,7 +442,7 @@ struct mem_ctl_info {
* If attributes are desired, then set to array of attributes * If attributes are desired, then set to array of attributes
* If no attributes are desired, leave NULL * If no attributes are desired, leave NULL
*/ */
struct mcidev_sysfs_attribute *mc_driver_sysfs_attributes; const struct mcidev_sysfs_attribute *mc_driver_sysfs_attributes;
/* work struct for this MC */ /* work struct for this MC */
struct delayed_work work; struct delayed_work work;
...@@ -810,6 +812,7 @@ extern struct mem_ctl_info *edac_mc_alloc(unsigned sz_pvt, unsigned nr_csrows, ...@@ -810,6 +812,7 @@ extern struct mem_ctl_info *edac_mc_alloc(unsigned sz_pvt, unsigned nr_csrows,
extern int edac_mc_add_mc(struct mem_ctl_info *mci); extern int edac_mc_add_mc(struct mem_ctl_info *mci);
extern void edac_mc_free(struct mem_ctl_info *mci); extern void edac_mc_free(struct mem_ctl_info *mci);
extern struct mem_ctl_info *edac_mc_find(int idx); extern struct mem_ctl_info *edac_mc_find(int idx);
extern struct mem_ctl_info *find_mci_by_dev(struct device *dev);
extern struct mem_ctl_info *edac_mc_del_mc(struct device *dev); extern struct mem_ctl_info *edac_mc_del_mc(struct device *dev);
extern int edac_mc_find_csrow_by_page(struct mem_ctl_info *mci, extern int edac_mc_find_csrow_by_page(struct mem_ctl_info *mci,
unsigned long page); unsigned long page);
......
...@@ -207,6 +207,7 @@ struct mem_ctl_info *edac_mc_alloc(unsigned sz_pvt, unsigned nr_csrows, ...@@ -207,6 +207,7 @@ struct mem_ctl_info *edac_mc_alloc(unsigned sz_pvt, unsigned nr_csrows,
} }
mci->op_state = OP_ALLOC; mci->op_state = OP_ALLOC;
INIT_LIST_HEAD(&mci->grp_kobj_list);
/* /*
* Initialize the 'root' kobj for the edac_mc controller * Initialize the 'root' kobj for the edac_mc controller
...@@ -234,18 +235,24 @@ EXPORT_SYMBOL_GPL(edac_mc_alloc); ...@@ -234,18 +235,24 @@ EXPORT_SYMBOL_GPL(edac_mc_alloc);
*/ */
void edac_mc_free(struct mem_ctl_info *mci) void edac_mc_free(struct mem_ctl_info *mci)
{ {
debugf1("%s()\n", __func__);
edac_mc_unregister_sysfs_main_kobj(mci); edac_mc_unregister_sysfs_main_kobj(mci);
/* free the mci instance memory here */
kfree(mci);
} }
EXPORT_SYMBOL_GPL(edac_mc_free); EXPORT_SYMBOL_GPL(edac_mc_free);
/* /**
* find_mci_by_dev * find_mci_by_dev
* *
* scan list of controllers looking for the one that manages * scan list of controllers looking for the one that manages
* the 'dev' device * the 'dev' device
* @dev: pointer to a struct device related with the MCI
*/ */
static struct mem_ctl_info *find_mci_by_dev(struct device *dev) struct mem_ctl_info *find_mci_by_dev(struct device *dev)
{ {
struct mem_ctl_info *mci; struct mem_ctl_info *mci;
struct list_head *item; struct list_head *item;
...@@ -261,6 +268,7 @@ static struct mem_ctl_info *find_mci_by_dev(struct device *dev) ...@@ -261,6 +268,7 @@ static struct mem_ctl_info *find_mci_by_dev(struct device *dev)
return NULL; return NULL;
} }
EXPORT_SYMBOL_GPL(find_mci_by_dev);
/* /*
* handler for EDAC to check if NMI type handler has asserted interrupt * handler for EDAC to check if NMI type handler has asserted interrupt
......
...@@ -631,9 +631,6 @@ static void edac_mci_control_release(struct kobject *kobj) ...@@ -631,9 +631,6 @@ static void edac_mci_control_release(struct kobject *kobj)
/* decrement the module ref count */ /* decrement the module ref count */
module_put(mci->owner); module_put(mci->owner);
/* free the mci instance memory here */
kfree(mci);
} }
static struct kobj_type ktype_mci = { static struct kobj_type ktype_mci = {
...@@ -713,6 +710,8 @@ int edac_mc_register_sysfs_main_kobj(struct mem_ctl_info *mci) ...@@ -713,6 +710,8 @@ int edac_mc_register_sysfs_main_kobj(struct mem_ctl_info *mci)
*/ */
void edac_mc_unregister_sysfs_main_kobj(struct mem_ctl_info *mci) void edac_mc_unregister_sysfs_main_kobj(struct mem_ctl_info *mci)
{ {
debugf1("%s()\n", __func__);
/* delete the kobj from the mc_kset */ /* delete the kobj from the mc_kset */
kobject_put(&mci->edac_mci_kobj); kobject_put(&mci->edac_mci_kobj);
} }
...@@ -760,8 +759,6 @@ static void edac_inst_grp_release(struct kobject *kobj) ...@@ -760,8 +759,6 @@ static void edac_inst_grp_release(struct kobject *kobj)
grp = container_of(kobj, struct mcidev_sysfs_group_kobj, kobj); grp = container_of(kobj, struct mcidev_sysfs_group_kobj, kobj);
mci = grp->mci; mci = grp->mci;
kobject_put(&mci->edac_mci_kobj);
} }
/* Intermediate show/store table */ /* Intermediate show/store table */
...@@ -784,7 +781,7 @@ static struct kobj_type ktype_inst_grp = { ...@@ -784,7 +781,7 @@ static struct kobj_type ktype_inst_grp = {
* object tree. * object tree.
*/ */
static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci, static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci,
struct mcidev_sysfs_attribute *sysfs_attrib, const struct mcidev_sysfs_attribute *sysfs_attrib,
struct kobject *kobj) struct kobject *kobj)
{ {
int err; int err;
...@@ -792,6 +789,7 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci, ...@@ -792,6 +789,7 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci,
debugf1("%s()\n", __func__); debugf1("%s()\n", __func__);
while (sysfs_attrib) { while (sysfs_attrib) {
debugf1("%s() sysfs_attrib = %p\n",__func__, sysfs_attrib);
if (sysfs_attrib->grp) { if (sysfs_attrib->grp) {
struct mcidev_sysfs_group_kobj *grp_kobj; struct mcidev_sysfs_group_kobj *grp_kobj;
...@@ -799,10 +797,9 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci, ...@@ -799,10 +797,9 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci,
if (!grp_kobj) if (!grp_kobj)
return -ENOMEM; return -ENOMEM;
list_add_tail(&grp_kobj->list, &mci->grp_kobj_list);
grp_kobj->grp = sysfs_attrib->grp; grp_kobj->grp = sysfs_attrib->grp;
grp_kobj->mci = mci; grp_kobj->mci = mci;
list_add_tail(&grp_kobj->list, &mci->grp_kobj_list);
debugf0("%s() grp %s, mci %p\n", __func__, debugf0("%s() grp %s, mci %p\n", __func__,
sysfs_attrib->grp->name, mci); sysfs_attrib->grp->name, mci);
...@@ -811,26 +808,28 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci, ...@@ -811,26 +808,28 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci,
&ktype_inst_grp, &ktype_inst_grp,
&mci->edac_mci_kobj, &mci->edac_mci_kobj,
sysfs_attrib->grp->name); sysfs_attrib->grp->name);
if (err) if (err < 0) {
printk(KERN_ERR "kobject_init_and_add failed: %d\n", err);
return err; return err;
}
err = edac_create_mci_instance_attributes(mci, err = edac_create_mci_instance_attributes(mci,
grp_kobj->grp->mcidev_attr, grp_kobj->grp->mcidev_attr,
&grp_kobj->kobj); &grp_kobj->kobj);
if (err) if (err < 0)
return err; return err;
} else if (sysfs_attrib->attr.name) { } else if (sysfs_attrib->attr.name) {
debugf0("%s() file %s\n", __func__, debugf0("%s() file %s\n", __func__,
sysfs_attrib->attr.name); sysfs_attrib->attr.name);
err = sysfs_create_file(kobj, &sysfs_attrib->attr); err = sysfs_create_file(kobj, &sysfs_attrib->attr);
if (err < 0) {
printk(KERN_ERR "sysfs_create_file failed: %d\n", err);
return err;
}
} else } else
break; break;
if (err) {
return err;
}
sysfs_attrib++; sysfs_attrib++;
} }
...@@ -843,7 +842,7 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci, ...@@ -843,7 +842,7 @@ static int edac_create_mci_instance_attributes(struct mem_ctl_info *mci,
* directory of this mci instance. * directory of this mci instance.
*/ */
static void edac_remove_mci_instance_attributes(struct mem_ctl_info *mci, static void edac_remove_mci_instance_attributes(struct mem_ctl_info *mci,
struct mcidev_sysfs_attribute *sysfs_attrib, const struct mcidev_sysfs_attribute *sysfs_attrib,
struct kobject *kobj, int count) struct kobject *kobj, int count)
{ {
struct mcidev_sysfs_group_kobj *grp_kobj, *tmp; struct mcidev_sysfs_group_kobj *grp_kobj, *tmp;
...@@ -855,13 +854,24 @@ static void edac_remove_mci_instance_attributes(struct mem_ctl_info *mci, ...@@ -855,13 +854,24 @@ static void edac_remove_mci_instance_attributes(struct mem_ctl_info *mci,
* Remove first all the atributes * Remove first all the atributes
*/ */
while (sysfs_attrib) { while (sysfs_attrib) {
debugf1("%s() sysfs_attrib = %p\n",__func__, sysfs_attrib);
if (sysfs_attrib->grp) { if (sysfs_attrib->grp) {
list_for_each_entry(grp_kobj, &mci->grp_kobj_list, debugf1("%s() seeking for group %s\n",
list) __func__, sysfs_attrib->grp->name);
if (grp_kobj->grp == sysfs_attrib->grp) list_for_each_entry(grp_kobj,
&mci->grp_kobj_list, list) {
debugf1("%s() grp_kobj->grp = %p\n",__func__, grp_kobj->grp);
if (grp_kobj->grp == sysfs_attrib->grp) {
edac_remove_mci_instance_attributes(mci, edac_remove_mci_instance_attributes(mci,
grp_kobj->grp->mcidev_attr, grp_kobj->grp->mcidev_attr,
&grp_kobj->kobj, count + 1); &grp_kobj->kobj, count + 1);
debugf0("%s() group %s\n", __func__,
sysfs_attrib->grp->name);
kobject_put(&grp_kobj->kobj);
}
}
debugf1("%s() end of seeking for group %s\n",
__func__, sysfs_attrib->grp->name);
} else if (sysfs_attrib->attr.name) { } else if (sysfs_attrib->attr.name) {
debugf0("%s() file %s\n", __func__, debugf0("%s() file %s\n", __func__,
sysfs_attrib->attr.name); sysfs_attrib->attr.name);
...@@ -871,15 +881,14 @@ static void edac_remove_mci_instance_attributes(struct mem_ctl_info *mci, ...@@ -871,15 +881,14 @@ static void edac_remove_mci_instance_attributes(struct mem_ctl_info *mci,
sysfs_attrib++; sysfs_attrib++;
} }
/* /* Remove the group objects */
* Now that all attributes got removed, it is save to remove all groups if (count)
*/ return;
if (!count) list_for_each_entry_safe(grp_kobj, tmp,
list_for_each_entry_safe(grp_kobj, tmp, &mci->grp_kobj_list, &mci->grp_kobj_list, list) {
list) { list_del(&grp_kobj->list);
debugf0("%s() grp %s\n", __func__, grp_kobj->grp->name); kfree(grp_kobj);
kobject_put(&grp_kobj->kobj); }
}
} }
...@@ -971,6 +980,7 @@ void edac_remove_sysfs_mci_device(struct mem_ctl_info *mci) ...@@ -971,6 +980,7 @@ void edac_remove_sysfs_mci_device(struct mem_ctl_info *mci)
debugf0("%s()\n", __func__); debugf0("%s()\n", __func__);
/* remove all csrow kobjects */ /* remove all csrow kobjects */
debugf0("%s() unregister this mci kobj\n", __func__);
for (i = 0; i < mci->nr_csrows; i++) { for (i = 0; i < mci->nr_csrows; i++) {
if (mci->csrows[i].nr_pages > 0) { if (mci->csrows[i].nr_pages > 0) {
debugf0("%s() unreg csrow-%d\n", __func__, i); debugf0("%s() unreg csrow-%d\n", __func__, i);
...@@ -978,20 +988,20 @@ void edac_remove_sysfs_mci_device(struct mem_ctl_info *mci) ...@@ -978,20 +988,20 @@ void edac_remove_sysfs_mci_device(struct mem_ctl_info *mci)
} }
} }
debugf0("%s() remove_link\n", __func__); /* remove this mci instance's attribtes */
if (mci->mc_driver_sysfs_attributes) {
debugf0("%s() unregister mci private attributes\n", __func__);
edac_remove_mci_instance_attributes(mci,
mci->mc_driver_sysfs_attributes,
&mci->edac_mci_kobj, 0);
}
/* remove the symlink */ /* remove the symlink */
debugf0("%s() remove_link\n", __func__);
sysfs_remove_link(&mci->edac_mci_kobj, EDAC_DEVICE_SYMLINK); sysfs_remove_link(&mci->edac_mci_kobj, EDAC_DEVICE_SYMLINK);
debugf0("%s() remove_mci_instance\n", __func__);
/* remove this mci instance's attribtes */
edac_remove_mci_instance_attributes(mci,
mci->mc_driver_sysfs_attributes,
&mci->edac_mci_kobj, 0);
debugf0("%s() unregister this mci kobj\n", __func__);
/* unregister this instance's kobject */ /* unregister this instance's kobject */
debugf0("%s() remove_mci_instance\n", __func__);
kobject_put(&mci->edac_mci_kobj); kobject_put(&mci->edac_mci_kobj);
} }
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment