diff options
| author | Henrique de Moraes Holschuh <hmh@hmh.eng.br> | 2007-04-24 10:48:14 -0400 |
|---|---|---|
| committer | Len Brown <len.brown@intel.com> | 2007-04-25 02:00:27 -0400 |
| commit | 7252374a39d794879f5e47bcfa0a16e7599b27b5 (patch) | |
| tree | f6a0801099b74b9d78ea1240d725b7aa2f6c6a0b | |
| parent | 176750d68801bfa4a88d1cf54174aa0347d7e5d8 (diff) | |
ACPI: thinkpad-acpi: add infrastructure for the sysfs device attributes
Add infrastructure to deal with sysfs attributes and grouping, and helpers
for common sysfs parsing. Switch driver attributes to use them.
Signed-off-by: Henrique de Moraes Holschuh <hmh@hmh.eng.br>
Signed-off-by: Len Brown <len.brown@intel.com>
| -rw-r--r-- | drivers/misc/thinkpad_acpi.c | 86 | ||||
| -rw-r--r-- | drivers/misc/thinkpad_acpi.h | 21 |
2 files changed, 102 insertions, 5 deletions
diff --git a/drivers/misc/thinkpad_acpi.c b/drivers/misc/thinkpad_acpi.c index a31d00d570cb..ca6d15cdc5f0 100644 --- a/drivers/misc/thinkpad_acpi.c +++ b/drivers/misc/thinkpad_acpi.c | |||
| @@ -520,12 +520,8 @@ static ssize_t tpacpi_driver_debug_store(struct device_driver *drv, | |||
| 520 | const char *buf, size_t count) | 520 | const char *buf, size_t count) |
| 521 | { | 521 | { |
| 522 | unsigned long t; | 522 | unsigned long t; |
| 523 | char *endp; | ||
| 524 | 523 | ||
| 525 | t = simple_strtoul(buf, &endp, 0); | 524 | if (parse_strtoul(buf, 0xffff, &t)) |
| 526 | while (*endp && isspace(*endp)) | ||
| 527 | endp++; | ||
| 528 | if (*endp) | ||
| 529 | return -EINVAL; | 525 | return -EINVAL; |
| 530 | 526 | ||
| 531 | dbg_level = t; | 527 | dbg_level = t; |
| @@ -575,6 +571,86 @@ static void tpacpi_remove_driver_attributes(struct device_driver *drv) | |||
| 575 | driver_remove_file(drv, tpacpi_driver_attributes[i]); | 571 | driver_remove_file(drv, tpacpi_driver_attributes[i]); |
| 576 | } | 572 | } |
| 577 | 573 | ||
| 574 | /************************************************************************* | ||
| 575 | * sysfs support helpers | ||
| 576 | */ | ||
| 577 | |||
| 578 | struct attribute_set_obj { | ||
| 579 | struct attribute_set s; | ||
| 580 | struct attribute *a; | ||
| 581 | } __attribute__((packed)); | ||
| 582 | |||
| 583 | static struct attribute_set *create_attr_set(unsigned int max_members, | ||
| 584 | const char* name) | ||
| 585 | { | ||
| 586 | struct attribute_set_obj *sobj; | ||
| 587 | |||
| 588 | if (max_members == 0) | ||
| 589 | return NULL; | ||
| 590 | |||
| 591 | /* Allocates space for implicit NULL at the end too */ | ||
| 592 | sobj = kzalloc(sizeof(struct attribute_set_obj) + | ||
| 593 | max_members * sizeof(struct attribute *), | ||
| 594 | GFP_KERNEL); | ||
| 595 | if (!sobj) | ||
| 596 | return NULL; | ||
| 597 | sobj->s.max_members = max_members; | ||
| 598 | sobj->s.group.attrs = &sobj->a; | ||
| 599 | sobj->s.group.name = name; | ||
| 600 | |||
| 601 | return &sobj->s; | ||
| 602 | } | ||
| 603 | |||
| 604 | /* not multi-threaded safe, use it in a single thread per set */ | ||
| 605 | static int add_to_attr_set(struct attribute_set* s, struct attribute *attr) | ||
| 606 | { | ||
| 607 | if (!s || !attr) | ||
| 608 | return -EINVAL; | ||
| 609 | |||
| 610 | if (s->members >= s->max_members) | ||
| 611 | return -ENOMEM; | ||
| 612 | |||
| 613 | s->group.attrs[s->members] = attr; | ||
| 614 | s->members++; | ||
| 615 | |||
| 616 | return 0; | ||
| 617 | } | ||
| 618 | |||
| 619 | static int add_many_to_attr_set(struct attribute_set* s, | ||
| 620 | struct attribute **attr, | ||
| 621 | unsigned int count) | ||
| 622 | { | ||
| 623 | int i, res; | ||
| 624 | |||
| 625 | for (i = 0; i < count; i++) { | ||
| 626 | res = add_to_attr_set(s, attr[i]); | ||
| 627 | if (res) | ||
| 628 | return res; | ||
| 629 | } | ||
| 630 | |||
| 631 | return 0; | ||
| 632 | } | ||
| 633 | |||
| 634 | static void delete_attr_set(struct attribute_set* s, struct kobject *kobj) | ||
| 635 | { | ||
| 636 | sysfs_remove_group(kobj, &s->group); | ||
| 637 | destroy_attr_set(s); | ||
| 638 | } | ||
| 639 | |||
| 640 | static int parse_strtoul(const char *buf, | ||
| 641 | unsigned long max, unsigned long *value) | ||
| 642 | { | ||
| 643 | char *endp; | ||
| 644 | |||
| 645 | *value = simple_strtoul(buf, &endp, 0); | ||
| 646 | while (*endp && isspace(*endp)) | ||
| 647 | endp++; | ||
| 648 | if (*endp || *value > max) | ||
| 649 | return -EINVAL; | ||
| 650 | |||
| 651 | return 0; | ||
| 652 | } | ||
| 653 | |||
| 578 | /**************************************************************************** | 654 | /**************************************************************************** |
| 579 | **************************************************************************** | 655 | **************************************************************************** |
| 580 | * | 656 | * |
diff --git a/drivers/misc/thinkpad_acpi.h b/drivers/misc/thinkpad_acpi.h index 37860582956f..84fdefe0d200 100644 --- a/drivers/misc/thinkpad_acpi.h +++ b/drivers/misc/thinkpad_acpi.h | |||
| @@ -134,6 +134,27 @@ static int dispatch_procfs_write(struct file *file, | |||
| 134 | unsigned long count, void *data); | 134 | unsigned long count, void *data); |
| 135 | static char *next_cmd(char **cmds); | 135 | static char *next_cmd(char **cmds); |
| 136 | 136 | ||
| 137 | /* sysfs support */ | ||
| 138 | struct attribute_set { | ||
| 139 | unsigned int members, max_members; | ||
| 140 | struct attribute_group group; | ||
| 141 | }; | ||
| 142 | |||
| 143 | static struct attribute_set *create_attr_set(unsigned int max_members, | ||
| 144 | const char* name); | ||
| 145 | #define destroy_attr_set(_set) \ | ||
| 146 | kfree(_set); | ||
| 147 | static int add_to_attr_set(struct attribute_set* s, struct attribute *attr); | ||
| 148 | static int add_many_to_attr_set(struct attribute_set* s, | ||
| 149 | struct attribute **attr, | ||
| 150 | unsigned int count); | ||
| 151 | #define register_attr_set_with_sysfs(_attr_set, _kobj) \ | ||
| 152 | sysfs_create_group(_kobj, &_attr_set->group) | ||
| 153 | static void delete_attr_set(struct attribute_set* s, struct kobject *kobj); | ||
| 154 | |||
| 155 | static int parse_strtoul(const char *buf, unsigned long max, | ||
| 156 | unsigned long *value); | ||
| 157 | |||
| 137 | /* Device model */ | 158 | /* Device model */ |
| 138 | static struct platform_device *tpacpi_pdev; | 159 | static struct platform_device *tpacpi_pdev; |
| 139 | static struct class_device *tpacpi_hwmon; | 160 | static struct class_device *tpacpi_hwmon; |
