diff options
| author | Peter Oberparleiter <peter.oberparleiter@de.ibm.com> | 2006-09-20 09:59:49 -0400 |
|---|---|---|
| committer | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2006-09-20 09:59:49 -0400 |
| commit | db0c2d59087296b3567ec408abe17108db88b385 (patch) | |
| tree | 303981d2ff2b5a10069fbee1692189f1b1739608 | |
| parent | 250b2dc83347feb73eb6bdf7511685e72b587e68 (diff) | |
[S390] set modalias for ccw bus uevents.
Add the MODALIAS environment variable for ccw bus uevents.
Signed-off-by: Peter Oberparleiter <peter.oberparleiter@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
| -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 |
