aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKevin Cernekee <kpc.mtd@gmail.com>2009-04-03 16:00:45 -0400
committerDavid Woodhouse <David.Woodhouse@intel.com>2009-04-04 09:32:39 -0400
commit694bb7fc19c6b87e106f4c85a2707072e2f111a0 (patch)
tree7c5e2f4cfbe89fee0779b8d7d92b51a84a276747
parent1f24b5a8ecbb2a3c7080f418974d40e3ffedb221 (diff)
[MTD] driver model updates (part 2)
1) Add more sysfs attributes: flags, size, erasesize, writesize, oobsize, numeraseregions, name 2) Move core_initcall() code into init_mtd(). The original approach does not work if CONFIG_MTD=m . 3) Add device_unregister() in del_mtd_device() so that devices get removed from sysfs as each driver is unloaded. Signed-off-by: Kevin Cernekee <kpc.mtd@gmail.com> Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
-rw-r--r--drivers/mtd/mtdcore.c111
1 files changed, 90 insertions, 21 deletions
diff --git a/drivers/mtd/mtdcore.c b/drivers/mtd/mtdcore.c
index a88f8bc9a534..89c1e5ddc7c2 100644
--- a/drivers/mtd/mtdcore.c
+++ b/drivers/mtd/mtdcore.c
@@ -89,11 +89,89 @@ static ssize_t mtd_type_show(struct device *dev,
89 89
90 return snprintf(buf, PAGE_SIZE, "%s\n", type); 90 return snprintf(buf, PAGE_SIZE, "%s\n", type);
91} 91}
92static DEVICE_ATTR(mtd_type, S_IRUGO, mtd_type_show, NULL); 92static DEVICE_ATTR(type, S_IRUGO, mtd_type_show, NULL);
93
94static ssize_t mtd_flags_show(struct device *dev,
95 struct device_attribute *attr, char *buf)
96{
97 struct mtd_info *mtd = dev_to_mtd(dev);
98
99 return snprintf(buf, PAGE_SIZE, "0x%lx\n", (unsigned long)mtd->flags);
100
101}
102static DEVICE_ATTR(flags, S_IRUGO, mtd_flags_show, NULL);
103
104static ssize_t mtd_size_show(struct device *dev,
105 struct device_attribute *attr, char *buf)
106{
107 struct mtd_info *mtd = dev_to_mtd(dev);
108
109 return snprintf(buf, PAGE_SIZE, "%llu\n",
110 (unsigned long long)mtd->size);
111
112}
113static DEVICE_ATTR(size, S_IRUGO, mtd_size_show, NULL);
114
115static ssize_t mtd_erasesize_show(struct device *dev,
116 struct device_attribute *attr, char *buf)
117{
118 struct mtd_info *mtd = dev_to_mtd(dev);
119
120 return snprintf(buf, PAGE_SIZE, "%lu\n", (unsigned long)mtd->erasesize);
121
122}
123static DEVICE_ATTR(erasesize, S_IRUGO, mtd_erasesize_show, NULL);
124
125static ssize_t mtd_writesize_show(struct device *dev,
126 struct device_attribute *attr, char *buf)
127{
128 struct mtd_info *mtd = dev_to_mtd(dev);
129
130 return snprintf(buf, PAGE_SIZE, "%lu\n", (unsigned long)mtd->writesize);
131
132}
133static DEVICE_ATTR(writesize, S_IRUGO, mtd_writesize_show, NULL);
134
135static ssize_t mtd_oobsize_show(struct device *dev,
136 struct device_attribute *attr, char *buf)
137{
138 struct mtd_info *mtd = dev_to_mtd(dev);
139
140 return snprintf(buf, PAGE_SIZE, "%lu\n", (unsigned long)mtd->oobsize);
141
142}
143static DEVICE_ATTR(oobsize, S_IRUGO, mtd_oobsize_show, NULL);
144
145static ssize_t mtd_numeraseregions_show(struct device *dev,
146 struct device_attribute *attr, char *buf)
147{
148 struct mtd_info *mtd = dev_to_mtd(dev);
149
150 return snprintf(buf, PAGE_SIZE, "%u\n", mtd->numeraseregions);
151
152}
153static DEVICE_ATTR(numeraseregions, S_IRUGO, mtd_numeraseregions_show,
154 NULL);
155
156static ssize_t mtd_name_show(struct device *dev,
157 struct device_attribute *attr, char *buf)
158{
159 struct mtd_info *mtd = dev_to_mtd(dev);
160
161 return snprintf(buf, PAGE_SIZE, "%s\n", mtd->name);
162
163}
164static DEVICE_ATTR(name, S_IRUGO, mtd_name_show, NULL);
93 165
94static struct attribute *mtd_attrs[] = { 166static struct attribute *mtd_attrs[] = {
95 &dev_attr_mtd_type.attr, 167 &dev_attr_type.attr,
96 /* FIXME provide a /proc/mtd superset */ 168 &dev_attr_flags.attr,
169 &dev_attr_size.attr,
170 &dev_attr_erasesize.attr,
171 &dev_attr_writesize.attr,
172 &dev_attr_oobsize.attr,
173 &dev_attr_numeraseregions.attr,
174 &dev_attr_name.attr,
97 NULL, 175 NULL,
98}; 176};
99 177
@@ -236,6 +314,8 @@ int del_mtd_device (struct mtd_info *mtd)
236 } else { 314 } else {
237 struct mtd_notifier *not; 315 struct mtd_notifier *not;
238 316
317 device_unregister(&mtd->dev);
318
239 /* No need to get a refcount on the module containing 319 /* No need to get a refcount on the module containing
240 the notifier, since we hold the mtd_table_mutex */ 320 the notifier, since we hold the mtd_table_mutex */
241 list_for_each_entry(not, &mtd_notifiers, list) 321 list_for_each_entry(not, &mtd_notifiers, list)
@@ -455,24 +535,6 @@ EXPORT_SYMBOL_GPL(register_mtd_user);
455EXPORT_SYMBOL_GPL(unregister_mtd_user); 535EXPORT_SYMBOL_GPL(unregister_mtd_user);
456EXPORT_SYMBOL_GPL(default_mtd_writev); 536EXPORT_SYMBOL_GPL(default_mtd_writev);
457 537
458static int __init mtd_setup(void)
459{
460 mtd_class = class_create(THIS_MODULE, "mtd");
461
462 if (IS_ERR(mtd_class)) {
463 pr_err("Error creating mtd class.\n");
464 return PTR_ERR(mtd_class);
465 }
466 return 0;
467}
468core_initcall(mtd_setup);
469
470static void __exit mtd_teardown(void)
471{
472 class_destroy(mtd_class);
473}
474__exitcall(mtd_teardown);
475
476#ifdef CONFIG_PROC_FS 538#ifdef CONFIG_PROC_FS
477 539
478/*====================================================================*/ 540/*====================================================================*/
@@ -528,6 +590,12 @@ done:
528 590
529static int __init init_mtd(void) 591static int __init init_mtd(void)
530{ 592{
593 mtd_class = class_create(THIS_MODULE, "mtd");
594
595 if (IS_ERR(mtd_class)) {
596 pr_err("Error creating mtd class.\n");
597 return PTR_ERR(mtd_class);
598 }
531 if ((proc_mtd = create_proc_entry( "mtd", 0, NULL ))) 599 if ((proc_mtd = create_proc_entry( "mtd", 0, NULL )))
532 proc_mtd->read_proc = mtd_read_proc; 600 proc_mtd->read_proc = mtd_read_proc;
533 return 0; 601 return 0;
@@ -537,6 +605,7 @@ static void __exit cleanup_mtd(void)
537{ 605{
538 if (proc_mtd) 606 if (proc_mtd)
539 remove_proc_entry( "mtd", NULL); 607 remove_proc_entry( "mtd", NULL);
608 class_destroy(mtd_class);
540} 609}
541 610
542module_init(init_mtd); 611module_init(init_mtd);