diff options
Diffstat (limited to 'drivers/s390/cio')
-rw-r--r-- | drivers/s390/cio/device.c | 109 |
1 files changed, 66 insertions, 43 deletions
diff --git a/drivers/s390/cio/device.c b/drivers/s390/cio/device.c index 646da5640401..688945662c15 100644 --- a/drivers/s390/cio/device.c +++ b/drivers/s390/cio/device.c | |||
@@ -52,53 +52,81 @@ ccw_bus_match (struct device * dev, struct device_driver * drv) | |||
52 | return 1; | 52 | return 1; |
53 | } | 53 | } |
54 | 54 | ||
55 | /* | 55 | /* Store modalias string delimited by prefix/suffix string into buffer with |
56 | * Hotplugging interface for ccw devices. | 56 | * specified size. Return length of resulting string (excluding trailing '\0') |
57 | * Heavily modeled on pci and usb hotplug. | 57 | * even if string doesn't fit buffer (snprintf semantics). */ |
58 | */ | 58 | static int snprint_alias(char *buf, size_t size, const char *prefix, |
59 | static int | 59 | struct ccw_device_id *id, const char *suffix) |
60 | ccw_uevent (struct device *dev, char **envp, int num_envp, | ||
61 | char *buffer, int buffer_size) | ||
62 | { | 60 | { |
63 | struct ccw_device *cdev = to_ccwdev(dev); | 61 | int len; |
64 | int i = 0; | ||
65 | int length = 0; | ||
66 | 62 | ||
67 | if (!cdev) | 63 | len = snprintf(buf, size, "%sccw:t%04Xm%02X", prefix, id->cu_type, |
68 | return -ENODEV; | 64 | id->cu_model); |
65 | if (len > size) | ||
66 | return len; | ||
67 | buf += len; | ||
68 | size -= len; | ||
69 | 69 | ||
70 | /* what we want to pass to /sbin/hotplug */ | 70 | if (id->dev_type != 0) |
71 | len += snprintf(buf, size, "dt%04Xdm%02X%s", id->dev_type, | ||
72 | id->dev_model, suffix); | ||
73 | else | ||
74 | len += snprintf(buf, size, "dtdm%s", suffix); | ||
71 | 75 | ||
72 | envp[i++] = buffer; | 76 | return len; |
73 | length += scnprintf(buffer, buffer_size - length, "CU_TYPE=%04X", | 77 | } |
74 | cdev->id.cu_type); | ||
75 | if ((buffer_size - length <= 0) || (i >= num_envp)) | ||
76 | return -ENOMEM; | ||
77 | ++length; | ||
78 | buffer += length; | ||
79 | 78 | ||
79 | /* Set up environment variables for ccw device uevent. Return 0 on success, | ||
80 | * non-zero otherwise. */ | ||
81 | static int ccw_uevent(struct device *dev, char **envp, int num_envp, | ||
82 | char *buffer, int buffer_size) | ||
83 | { | ||
84 | struct ccw_device *cdev = to_ccwdev(dev); | ||
85 | struct ccw_device_id *id = &(cdev->id); | ||
86 | int i = 0; | ||
87 | int len; | ||
88 | |||
89 | /* CU_TYPE= */ | ||
90 | len = snprintf(buffer, buffer_size, "CU_TYPE=%04X", id->cu_type) + 1; | ||
91 | if (len > buffer_size || i >= num_envp) | ||
92 | return -ENOMEM; | ||
80 | envp[i++] = buffer; | 93 | envp[i++] = buffer; |
81 | length += scnprintf(buffer, buffer_size - length, "CU_MODEL=%02X", | 94 | buffer += len; |
82 | cdev->id.cu_model); | 95 | buffer_size -= len; |
83 | if ((buffer_size - length <= 0) || (i >= num_envp)) | 96 | |
97 | /* CU_MODEL= */ | ||
98 | len = snprintf(buffer, buffer_size, "CU_MODEL=%02X", id->cu_model) + 1; | ||
99 | if (len > buffer_size || i >= num_envp) | ||
84 | return -ENOMEM; | 100 | return -ENOMEM; |
85 | ++length; | 101 | envp[i++] = buffer; |
86 | buffer += length; | 102 | buffer += len; |
103 | buffer_size -= len; | ||
87 | 104 | ||
88 | /* The next two can be zero, that's ok for us */ | 105 | /* The next two can be zero, that's ok for us */ |
89 | envp[i++] = buffer; | 106 | /* DEV_TYPE= */ |
90 | length += scnprintf(buffer, buffer_size - length, "DEV_TYPE=%04X", | 107 | len = snprintf(buffer, buffer_size, "DEV_TYPE=%04X", id->dev_type) + 1; |
91 | cdev->id.dev_type); | 108 | if (len > buffer_size || i >= num_envp) |
92 | if ((buffer_size - length <= 0) || (i >= num_envp)) | ||
93 | return -ENOMEM; | 109 | return -ENOMEM; |
94 | ++length; | 110 | envp[i++] = buffer; |
95 | buffer += length; | 111 | buffer += len; |
112 | buffer_size -= len; | ||
96 | 113 | ||
114 | /* DEV_MODEL= */ | ||
115 | len = snprintf(buffer, buffer_size, "DEV_MODEL=%02X", | ||
116 | (unsigned char) id->dev_model) + 1; | ||
117 | if (len > buffer_size || i >= num_envp) | ||
118 | return -ENOMEM; | ||
97 | envp[i++] = buffer; | 119 | envp[i++] = buffer; |
98 | length += scnprintf(buffer, buffer_size - length, "DEV_MODEL=%02X", | 120 | buffer += len; |
99 | cdev->id.dev_model); | 121 | buffer_size -= len; |
100 | if ((buffer_size - length <= 0) || (i >= num_envp)) | 122 | |
123 | /* MODALIAS= */ | ||
124 | len = snprint_alias(buffer, buffer_size, "MODALIAS=", id, "") + 1; | ||
125 | if (len > buffer_size || i >= num_envp) | ||
101 | return -ENOMEM; | 126 | return -ENOMEM; |
127 | envp[i++] = buffer; | ||
128 | buffer += len; | ||
129 | buffer_size -= len; | ||
102 | 130 | ||
103 | envp[i] = NULL; | 131 | envp[i] = NULL; |
104 | 132 | ||
@@ -251,16 +279,11 @@ modalias_show (struct device *dev, struct device_attribute *attr, char *buf) | |||
251 | { | 279 | { |
252 | struct ccw_device *cdev = to_ccwdev(dev); | 280 | struct ccw_device *cdev = to_ccwdev(dev); |
253 | struct ccw_device_id *id = &(cdev->id); | 281 | struct ccw_device_id *id = &(cdev->id); |
254 | int ret; | 282 | int len; |
255 | 283 | ||
256 | ret = sprintf(buf, "ccw:t%04Xm%02X", | 284 | len = snprint_alias(buf, PAGE_SIZE, "", id, "\n") + 1; |
257 | id->cu_type, id->cu_model); | 285 | |
258 | if (id->dev_type != 0) | 286 | return len > PAGE_SIZE ? PAGE_SIZE : len; |
259 | ret += sprintf(buf + ret, "dt%04Xdm%02X\n", | ||
260 | id->dev_type, id->dev_model); | ||
261 | else | ||
262 | ret += sprintf(buf + ret, "dtdm\n"); | ||
263 | return ret; | ||
264 | } | 287 | } |
265 | 288 | ||
266 | static ssize_t | 289 | static ssize_t |