diff options
| -rw-r--r-- | drivers/usb/core/sysfs.c | 219 | ||||
| -rw-r--r-- | include/linux/usb.h | 5 |
2 files changed, 125 insertions, 99 deletions
diff --git a/drivers/usb/core/sysfs.c b/drivers/usb/core/sysfs.c index 4bdbc9df6e03..f18317fb49ee 100644 --- a/drivers/usb/core/sysfs.c +++ b/drivers/usb/core/sysfs.c | |||
| @@ -23,43 +23,56 @@ | |||
| 23 | #include "usb.h" | 23 | #include "usb.h" |
| 24 | 24 | ||
| 25 | /* endpoint stuff */ | 25 | /* endpoint stuff */ |
| 26 | struct endpoint_attribute { | 26 | struct ep_object { |
| 27 | struct device_attribute dev_attr; | 27 | struct usb_endpoint_descriptor *desc; |
| 28 | struct usb_endpoint_descriptor *endpoint; | ||
| 29 | struct usb_device *udev; | 28 | struct usb_device *udev; |
| 29 | struct kobject kobj; | ||
| 30 | }; | 30 | }; |
| 31 | #define to_endpoint_attr(_dev_attr) \ | 31 | #define to_ep_object(_kobj) \ |
| 32 | container_of(_dev_attr, struct endpoint_attribute, dev_attr) | 32 | container_of(_kobj, struct ep_object, kobj) |
| 33 | 33 | ||
| 34 | #define usb_ep_attr(field, format_string) \ | 34 | struct ep_attribute { |
| 35 | static ssize_t show_ep_##field(struct device *dev, struct device_attribute *attr, \ | 35 | struct attribute attr; |
| 36 | char *buf) \ | 36 | ssize_t (*show)(struct usb_device *, |
| 37 | { \ | 37 | struct usb_endpoint_descriptor *, char *); |
| 38 | struct endpoint_attribute *endpoint_attr = to_endpoint_attr(attr); \ | 38 | }; |
| 39 | \ | 39 | #define to_ep_attribute(_attr) \ |
| 40 | return sprintf(buf, format_string, endpoint_attr->endpoint->field); \ | 40 | container_of(_attr, struct ep_attribute, attr) |
| 41 | } | 41 | |
| 42 | #define EP_ATTR(_name) \ | ||
| 43 | struct ep_attribute ep_##_name = { \ | ||
| 44 | .attr = {.name = #_name, .owner = THIS_MODULE, \ | ||
| 45 | .mode = S_IRUGO}, \ | ||
| 46 | .show = show_ep_##_name} | ||
| 47 | |||
| 48 | #define usb_ep_attr(field, format_string) \ | ||
| 49 | static ssize_t show_ep_##field(struct usb_device *udev, \ | ||
| 50 | struct usb_endpoint_descriptor *desc, \ | ||
| 51 | char *buf) \ | ||
| 52 | { \ | ||
| 53 | return sprintf(buf, format_string, desc->field); \ | ||
| 54 | } \ | ||
| 55 | static EP_ATTR(field); | ||
| 56 | |||
| 42 | usb_ep_attr(bLength, "%02x\n") | 57 | usb_ep_attr(bLength, "%02x\n") |
| 43 | usb_ep_attr(bDescriptorType, "%02x\n") | ||
| 44 | usb_ep_attr(bEndpointAddress, "%02x\n") | 58 | usb_ep_attr(bEndpointAddress, "%02x\n") |
| 45 | usb_ep_attr(bmAttributes, "%02x\n") | 59 | usb_ep_attr(bmAttributes, "%02x\n") |
| 46 | usb_ep_attr(bInterval, "%02x\n") | 60 | usb_ep_attr(bInterval, "%02x\n") |
| 47 | 61 | ||
| 48 | static ssize_t show_ep_wMaxPacketSize(struct device *dev, | 62 | static ssize_t show_ep_wMaxPacketSize(struct usb_device *udev, |
| 49 | struct device_attribute *attr, char *buf) | 63 | struct usb_endpoint_descriptor *desc, char *buf) |
| 50 | { | 64 | { |
| 51 | struct endpoint_attribute *endpoint_attr = to_endpoint_attr(attr); | ||
| 52 | |||
| 53 | return sprintf(buf, "%04x\n", | 65 | return sprintf(buf, "%04x\n", |
| 54 | le16_to_cpu(endpoint_attr->endpoint->wMaxPacketSize) & 0x07ff); | 66 | le16_to_cpu(desc->wMaxPacketSize) & 0x07ff); |
| 55 | } | 67 | } |
| 68 | static EP_ATTR(wMaxPacketSize); | ||
| 56 | 69 | ||
| 57 | static ssize_t show_ep_type(struct device *dev, struct device_attribute *attr, char *buf) | 70 | static ssize_t show_ep_type(struct usb_device *udev, |
| 71 | struct usb_endpoint_descriptor *desc, char *buf) | ||
| 58 | { | 72 | { |
| 59 | struct endpoint_attribute *endpoint_attr = to_endpoint_attr(attr); | ||
| 60 | char *type = "unknown"; | 73 | char *type = "unknown"; |
| 61 | 74 | ||
| 62 | switch (endpoint_attr->endpoint->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) { | 75 | switch (desc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) { |
| 63 | case USB_ENDPOINT_XFER_CONTROL: | 76 | case USB_ENDPOINT_XFER_CONTROL: |
| 64 | type = "Control"; | 77 | type = "Control"; |
| 65 | break; | 78 | break; |
| @@ -75,35 +88,34 @@ static ssize_t show_ep_type(struct device *dev, struct device_attribute *attr, c | |||
| 75 | } | 88 | } |
| 76 | return sprintf(buf, "%s\n", type); | 89 | return sprintf(buf, "%s\n", type); |
| 77 | } | 90 | } |
| 91 | static EP_ATTR(type); | ||
| 78 | 92 | ||
| 79 | static ssize_t show_ep_interval(struct device *dev, struct device_attribute *attr, char *buf) | 93 | static ssize_t show_ep_interval(struct usb_device *udev, |
| 94 | struct usb_endpoint_descriptor *desc, char *buf) | ||
| 80 | { | 95 | { |
| 81 | struct endpoint_attribute *endpoint_attr = to_endpoint_attr(attr); | ||
| 82 | struct usb_device *udev = endpoint_attr->udev; | ||
| 83 | struct usb_endpoint_descriptor *endpoint = endpoint_attr->endpoint; | ||
| 84 | char unit; | 96 | char unit; |
| 85 | unsigned interval = 0; | 97 | unsigned interval = 0; |
| 86 | unsigned in; | 98 | unsigned in; |
| 87 | 99 | ||
| 88 | in = (endpoint->bEndpointAddress & USB_DIR_IN); | 100 | in = (desc->bEndpointAddress & USB_DIR_IN); |
| 89 | 101 | ||
| 90 | switch (endpoint->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) { | 102 | switch (desc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) { |
| 91 | case USB_ENDPOINT_XFER_CONTROL: | 103 | case USB_ENDPOINT_XFER_CONTROL: |
| 92 | if (udev->speed == USB_SPEED_HIGH) /* uframes per NAK */ | 104 | if (udev->speed == USB_SPEED_HIGH) /* uframes per NAK */ |
| 93 | interval = endpoint->bInterval; | 105 | interval = desc->bInterval; |
| 94 | break; | 106 | break; |
| 95 | case USB_ENDPOINT_XFER_ISOC: | 107 | case USB_ENDPOINT_XFER_ISOC: |
| 96 | interval = 1 << (endpoint->bInterval - 1); | 108 | interval = 1 << (desc->bInterval - 1); |
| 97 | break; | 109 | break; |
| 98 | case USB_ENDPOINT_XFER_BULK: | 110 | case USB_ENDPOINT_XFER_BULK: |
| 99 | if (udev->speed == USB_SPEED_HIGH && !in) /* uframes per NAK */ | 111 | if (udev->speed == USB_SPEED_HIGH && !in) /* uframes per NAK */ |
| 100 | interval = endpoint->bInterval; | 112 | interval = desc->bInterval; |
| 101 | break; | 113 | break; |
| 102 | case USB_ENDPOINT_XFER_INT: | 114 | case USB_ENDPOINT_XFER_INT: |
| 103 | if (udev->speed == USB_SPEED_HIGH) { | 115 | if (udev->speed == USB_SPEED_HIGH) |
| 104 | interval = 1 << (endpoint->bInterval - 1); | 116 | interval = 1 << (desc->bInterval - 1); |
| 105 | } else | 117 | else |
| 106 | interval = endpoint->bInterval; | 118 | interval = desc->bInterval; |
| 107 | break; | 119 | break; |
| 108 | } | 120 | } |
| 109 | interval *= (udev->speed == USB_SPEED_HIGH) ? 125 : 1000; | 121 | interval *= (udev->speed == USB_SPEED_HIGH) ? 125 : 1000; |
| @@ -116,78 +128,95 @@ static ssize_t show_ep_interval(struct device *dev, struct device_attribute *att | |||
| 116 | 128 | ||
| 117 | return sprintf(buf, "%d%cs\n", interval, unit); | 129 | return sprintf(buf, "%d%cs\n", interval, unit); |
| 118 | } | 130 | } |
| 131 | static EP_ATTR(interval); | ||
| 119 | 132 | ||
| 120 | static ssize_t show_ep_direction(struct device *dev, struct device_attribute *attr, char *buf) | 133 | static ssize_t show_ep_direction(struct usb_device *udev, |
| 134 | struct usb_endpoint_descriptor *desc, char *buf) | ||
| 121 | { | 135 | { |
| 122 | struct endpoint_attribute *endpoint_attr = to_endpoint_attr(attr); | ||
| 123 | char *direction; | 136 | char *direction; |
| 124 | 137 | ||
| 125 | if ((endpoint_attr->endpoint->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) == | 138 | if ((desc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) == |
| 126 | USB_ENDPOINT_XFER_CONTROL) | 139 | USB_ENDPOINT_XFER_CONTROL) |
| 127 | direction = "both"; | 140 | direction = "both"; |
| 128 | else if (endpoint_attr->endpoint->bEndpointAddress & USB_DIR_IN) | 141 | else if (desc->bEndpointAddress & USB_DIR_IN) |
| 129 | direction = "in"; | 142 | direction = "in"; |
| 130 | else | 143 | else |
| 131 | direction = "out"; | 144 | direction = "out"; |
| 132 | return sprintf(buf, "%s\n", direction); | 145 | return sprintf(buf, "%s\n", direction); |
| 133 | } | 146 | } |
| 147 | static EP_ATTR(direction); | ||
| 148 | |||
| 149 | static struct attribute *ep_attrs[] = { | ||
| 150 | &ep_bLength.attr, | ||
| 151 | &ep_bEndpointAddress.attr, | ||
| 152 | &ep_bmAttributes.attr, | ||
| 153 | &ep_bInterval.attr, | ||
| 154 | &ep_wMaxPacketSize.attr, | ||
| 155 | &ep_type.attr, | ||
| 156 | &ep_interval.attr, | ||
| 157 | &ep_direction.attr, | ||
| 158 | NULL, | ||
| 159 | }; | ||
| 134 | 160 | ||
| 135 | static struct endpoint_attribute *create_ep_attr(struct usb_endpoint_descriptor *endpoint, | 161 | static void ep_object_release(struct kobject *kobj) |
| 136 | struct usb_device *udev, char *name, | ||
| 137 | ssize_t (*show)(struct device *dev, struct device_attribute *attr, char *buf)) | ||
| 138 | { | 162 | { |
| 139 | struct endpoint_attribute *ep_attr; | 163 | kfree(to_ep_object(kobj)); |
| 140 | |||
| 141 | ep_attr = kzalloc(sizeof(*ep_attr), GFP_KERNEL); | ||
| 142 | if (ep_attr) { | ||
| 143 | ep_attr->endpoint = endpoint; | ||
| 144 | ep_attr->udev = udev; | ||
| 145 | ep_attr->dev_attr.attr.name = name; | ||
| 146 | ep_attr->dev_attr.attr.mode = 0444; | ||
| 147 | ep_attr->dev_attr.attr.owner = THIS_MODULE; | ||
| 148 | ep_attr->dev_attr.show = show; | ||
| 149 | } | ||
| 150 | return ep_attr; | ||
| 151 | } | 164 | } |
| 152 | 165 | ||
| 153 | static void usb_create_ep_files(struct kobject *kobj, struct usb_host_endpoint *endpoint, struct usb_device *udev) | 166 | static ssize_t ep_object_show(struct kobject *kobj, struct attribute *attr, |
| 167 | char *buf) | ||
| 154 | { | 168 | { |
| 155 | struct usb_endpoint_descriptor *ep; | 169 | struct ep_object *ep_obj = to_ep_object(kobj); |
| 156 | 170 | struct ep_attribute *ep_attr = to_ep_attribute(attr); | |
| 157 | ep = &endpoint->desc; | 171 | |
| 158 | 172 | return (ep_attr->show)(ep_obj->udev, ep_obj->desc, buf); | |
| 159 | endpoint->attrs = kzalloc(sizeof(struct attribute *) * 10, GFP_KERNEL); | ||
| 160 | endpoint->attrs[0] = &(create_ep_attr(ep, udev, "direction", show_ep_direction)->dev_attr.attr); | ||
| 161 | endpoint->attrs[1] = &(create_ep_attr(ep, udev, "type", show_ep_type)->dev_attr.attr); | ||
| 162 | endpoint->attrs[2] = &(create_ep_attr(ep, udev, "bLength", show_ep_bLength)->dev_attr.attr); | ||
| 163 | endpoint->attrs[3] = &(create_ep_attr(ep, udev, "bDescriptorType", show_ep_bDescriptorType)->dev_attr.attr); | ||
| 164 | endpoint->attrs[4] = &(create_ep_attr(ep, udev, "bEndpointAddress", show_ep_bEndpointAddress)->dev_attr.attr); | ||
| 165 | endpoint->attrs[5] = &(create_ep_attr(ep, udev, "bmAttributes", show_ep_bmAttributes)->dev_attr.attr); | ||
| 166 | endpoint->attrs[6] = &(create_ep_attr(ep, udev, "wMaxPacketSize", show_ep_wMaxPacketSize)->dev_attr.attr); | ||
| 167 | endpoint->attrs[7] = &(create_ep_attr(ep, udev, "bInterval", show_ep_bInterval)->dev_attr.attr); | ||
| 168 | endpoint->attrs[8] = &(create_ep_attr(ep, udev, "interval", show_ep_interval)->dev_attr.attr); | ||
| 169 | endpoint->attrs[9] = NULL; | ||
| 170 | endpoint->num_attrs = 9; | ||
| 171 | |||
| 172 | endpoint->attr_group = kzalloc(sizeof(*endpoint->attr_group), GFP_KERNEL); | ||
| 173 | endpoint->attr_name = kzalloc(10, GFP_KERNEL); | ||
| 174 | sprintf(endpoint->attr_name, "ep_%02x", endpoint->desc.bEndpointAddress); | ||
| 175 | |||
| 176 | endpoint->attr_group->attrs = endpoint->attrs; | ||
| 177 | endpoint->attr_group->name = endpoint->attr_name; | ||
| 178 | sysfs_create_group(kobj, endpoint->attr_group); | ||
| 179 | } | 173 | } |
| 180 | 174 | ||
| 181 | static void usb_remove_ep_files(struct kobject *kobj, struct usb_host_endpoint *endpoint) | 175 | static struct sysfs_ops ep_object_sysfs_ops = { |
| 176 | .show = ep_object_show, | ||
| 177 | }; | ||
| 178 | |||
| 179 | static struct kobj_type ep_object_ktype = { | ||
| 180 | .release = ep_object_release, | ||
| 181 | .sysfs_ops = &ep_object_sysfs_ops, | ||
| 182 | .default_attrs = ep_attrs, | ||
| 183 | }; | ||
| 184 | |||
| 185 | static void usb_create_ep_files(struct kobject *parent, | ||
| 186 | struct usb_host_endpoint *endpoint, | ||
| 187 | struct usb_device *udev) | ||
| 182 | { | 188 | { |
| 183 | int i; | 189 | struct ep_object *ep_obj; |
| 190 | struct kobject *kobj; | ||
| 191 | |||
| 192 | ep_obj = kzalloc(sizeof(struct ep_object), GFP_KERNEL); | ||
| 193 | if (!ep_obj) | ||
| 194 | return; | ||
| 184 | 195 | ||
| 185 | sysfs_remove_group(kobj, endpoint->attr_group); | 196 | ep_obj->desc = &endpoint->desc; |
| 186 | kfree(endpoint->attr_group); | 197 | ep_obj->udev = udev; |
| 187 | kfree(endpoint->attr_name); | 198 | |
| 188 | for (i = 0; i < endpoint->num_attrs; ++i) | 199 | kobj = &ep_obj->kobj; |
| 189 | kfree(endpoint->attrs[i]); | 200 | kobject_set_name(kobj, "ep_%02x", endpoint->desc.bEndpointAddress); |
| 190 | kfree(endpoint->attrs); | 201 | kobj->parent = parent; |
| 202 | kobj->ktype = &ep_object_ktype; | ||
| 203 | |||
| 204 | /* Don't use kobject_register, because it generates a hotplug event */ | ||
| 205 | kobject_init(kobj); | ||
| 206 | if (kobject_add(kobj) == 0) | ||
| 207 | endpoint->kobj = kobj; | ||
| 208 | else | ||
| 209 | kobject_put(kobj); | ||
| 210 | } | ||
| 211 | |||
| 212 | static void usb_remove_ep_files(struct usb_host_endpoint *endpoint) | ||
| 213 | { | ||
| 214 | |||
| 215 | if (endpoint->kobj) { | ||
| 216 | kobject_del(endpoint->kobj); | ||
| 217 | kobject_put(endpoint->kobj); | ||
| 218 | endpoint->kobj = NULL; | ||
| 219 | } | ||
| 191 | } | 220 | } |
| 192 | 221 | ||
| 193 | /* Active configuration fields */ | 222 | /* Active configuration fields */ |
| @@ -411,7 +440,7 @@ void usb_remove_sysfs_dev_files (struct usb_device *udev) | |||
| 411 | { | 440 | { |
| 412 | struct device *dev = &udev->dev; | 441 | struct device *dev = &udev->dev; |
| 413 | 442 | ||
| 414 | usb_remove_ep_files(&dev->kobj, &udev->ep0); | 443 | usb_remove_ep_files(&udev->ep0); |
| 415 | sysfs_remove_group(&dev->kobj, &dev_attr_grp); | 444 | sysfs_remove_group(&dev->kobj, &dev_attr_grp); |
| 416 | 445 | ||
| 417 | if (udev->descriptor.iManufacturer) | 446 | if (udev->descriptor.iManufacturer) |
| @@ -496,7 +525,7 @@ static struct attribute_group intf_attr_grp = { | |||
| 496 | .attrs = intf_attrs, | 525 | .attrs = intf_attrs, |
| 497 | }; | 526 | }; |
| 498 | 527 | ||
| 499 | static void usb_create_intf_ep_files(struct usb_interface *intf) | 528 | static inline void usb_create_intf_ep_files(struct usb_interface *intf) |
| 500 | { | 529 | { |
| 501 | struct usb_host_interface *iface_desc; | 530 | struct usb_host_interface *iface_desc; |
| 502 | int i; | 531 | int i; |
| @@ -504,17 +533,17 @@ static void usb_create_intf_ep_files(struct usb_interface *intf) | |||
| 504 | iface_desc = intf->cur_altsetting; | 533 | iface_desc = intf->cur_altsetting; |
| 505 | for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) | 534 | for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) |
| 506 | usb_create_ep_files(&intf->dev.kobj, &iface_desc->endpoint[i], | 535 | usb_create_ep_files(&intf->dev.kobj, &iface_desc->endpoint[i], |
| 507 | interface_to_usbdev(intf)); | 536 | interface_to_usbdev(intf)); |
| 508 | } | 537 | } |
| 509 | 538 | ||
| 510 | static void usb_remove_intf_ep_files(struct usb_interface *intf) | 539 | static inline void usb_remove_intf_ep_files(struct usb_interface *intf) |
| 511 | { | 540 | { |
| 512 | struct usb_host_interface *iface_desc; | 541 | struct usb_host_interface *iface_desc; |
| 513 | int i; | 542 | int i; |
| 514 | 543 | ||
| 515 | iface_desc = intf->cur_altsetting; | 544 | iface_desc = intf->cur_altsetting; |
| 516 | for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) | 545 | for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) |
| 517 | usb_remove_ep_files(&intf->dev.kobj, &iface_desc->endpoint[i]); | 546 | usb_remove_ep_files(&iface_desc->endpoint[i]); |
| 518 | } | 547 | } |
| 519 | 548 | ||
| 520 | void usb_create_sysfs_intf_files (struct usb_interface *intf) | 549 | void usb_create_sysfs_intf_files (struct usb_interface *intf) |
diff --git a/include/linux/usb.h b/include/linux/usb.h index a2d923fd54f9..465ff4585ca5 100644 --- a/include/linux/usb.h +++ b/include/linux/usb.h | |||
| @@ -57,10 +57,7 @@ struct usb_host_endpoint { | |||
| 57 | struct usb_endpoint_descriptor desc; | 57 | struct usb_endpoint_descriptor desc; |
| 58 | struct list_head urb_list; | 58 | struct list_head urb_list; |
| 59 | void *hcpriv; | 59 | void *hcpriv; |
| 60 | char *attr_name; | 60 | struct kobject *kobj; /* For sysfs info */ |
| 61 | struct attribute_group *attr_group; | ||
| 62 | struct attribute **attrs; | ||
| 63 | int num_attrs; | ||
| 64 | 61 | ||
| 65 | unsigned char *extra; /* Extra descriptors */ | 62 | unsigned char *extra; /* Extra descriptors */ |
| 66 | int extralen; | 63 | int extralen; |
