]> git.karo-electronics.de Git - karo-tx-linux.git/blob - drivers/rapidio/rio-sysfs.c
Merge remote-tracking branch 'leds/for-next'
[karo-tx-linux.git] / drivers / rapidio / rio-sysfs.c
1 /*
2  * RapidIO sysfs attributes and support
3  *
4  * Copyright 2005 MontaVista Software, Inc.
5  * Matt Porter <mporter@kernel.crashing.org>
6  *
7  * This program is free software; you can redistribute  it and/or modify it
8  * under  the terms of  the GNU General  Public License as published by the
9  * Free Software Foundation;  either version 2 of the  License, or (at your
10  * option) any later version.
11  */
12
13 #include <linux/kernel.h>
14 #include <linux/rio.h>
15 #include <linux/rio_drv.h>
16 #include <linux/stat.h>
17 #include <linux/capability.h>
18
19 #include "rio.h"
20
21 /* Sysfs support */
22 #define rio_config_attr(field, format_string)                                   \
23 static ssize_t                                                          \
24 field##_show(struct device *dev, struct device_attribute *attr, char *buf)                      \
25 {                                                                       \
26         struct rio_dev *rdev = to_rio_dev(dev);                         \
27                                                                         \
28         return sprintf(buf, format_string, rdev->field);                \
29 }                                                                       \
30
31 rio_config_attr(did, "0x%04x\n");
32 rio_config_attr(vid, "0x%04x\n");
33 rio_config_attr(device_rev, "0x%08x\n");
34 rio_config_attr(asm_did, "0x%04x\n");
35 rio_config_attr(asm_vid, "0x%04x\n");
36 rio_config_attr(asm_rev, "0x%04x\n");
37 rio_config_attr(destid, "0x%04x\n");
38 rio_config_attr(hopcount, "0x%02x\n");
39
40 static ssize_t routes_show(struct device *dev, struct device_attribute *attr, char *buf)
41 {
42         struct rio_dev *rdev = to_rio_dev(dev);
43         char *str = buf;
44         int i;
45
46         for (i = 0; i < RIO_MAX_ROUTE_ENTRIES(rdev->net->hport->sys_size);
47                         i++) {
48                 if (rdev->rswitch->route_table[i] == RIO_INVALID_ROUTE)
49                         continue;
50                 str +=
51                     sprintf(str, "%04x %02x\n", i,
52                             rdev->rswitch->route_table[i]);
53         }
54
55         return (str - buf);
56 }
57
58 static ssize_t lprev_show(struct device *dev,
59                           struct device_attribute *attr, char *buf)
60 {
61         struct rio_dev *rdev = to_rio_dev(dev);
62
63         return sprintf(buf, "%s\n",
64                         (rdev->prev) ? rio_name(rdev->prev) : "root");
65 }
66
67 static ssize_t lnext_show(struct device *dev,
68                           struct device_attribute *attr, char *buf)
69 {
70         struct rio_dev *rdev = to_rio_dev(dev);
71         char *str = buf;
72         int i;
73
74         if (rdev->pef & RIO_PEF_SWITCH) {
75                 for (i = 0; i < RIO_GET_TOTAL_PORTS(rdev->swpinfo); i++) {
76                         if (rdev->rswitch->nextdev[i])
77                                 str += sprintf(str, "%s\n",
78                                         rio_name(rdev->rswitch->nextdev[i]));
79                         else
80                                 str += sprintf(str, "null\n");
81                 }
82         }
83
84         return str - buf;
85 }
86
87 static ssize_t modalias_show(struct device *dev,
88                              struct device_attribute *attr, char *buf)
89 {
90         struct rio_dev *rdev = to_rio_dev(dev);
91
92         return sprintf(buf, "rapidio:v%04Xd%04Xav%04Xad%04X\n",
93                        rdev->vid, rdev->did, rdev->asm_vid, rdev->asm_did);
94 }
95
96 struct device_attribute rio_dev_attrs[] = {
97         __ATTR_RO(did),
98         __ATTR_RO(vid),
99         __ATTR_RO(device_rev),
100         __ATTR_RO(asm_did),
101         __ATTR_RO(asm_vid),
102         __ATTR_RO(asm_rev),
103         __ATTR_RO(lprev),
104         __ATTR_RO(destid),
105         __ATTR_RO(modalias),
106         __ATTR_NULL,
107 };
108
109 static DEVICE_ATTR(routes, S_IRUGO, routes_show, NULL);
110 static DEVICE_ATTR(lnext, S_IRUGO, lnext_show, NULL);
111 static DEVICE_ATTR(hopcount, S_IRUGO, hopcount_show, NULL);
112
113 static ssize_t
114 rio_read_config(struct file *filp, struct kobject *kobj,
115                 struct bin_attribute *bin_attr,
116                 char *buf, loff_t off, size_t count)
117 {
118         struct rio_dev *dev =
119             to_rio_dev(container_of(kobj, struct device, kobj));
120         unsigned int size = 0x100;
121         loff_t init_off = off;
122         u8 *data = (u8 *) buf;
123
124         /* Several chips lock up trying to read undefined config space */
125         if (capable(CAP_SYS_ADMIN))
126                 size = RIO_MAINT_SPACE_SZ;
127
128         if (off >= size)
129                 return 0;
130         if (off + count > size) {
131                 size -= off;
132                 count = size;
133         } else {
134                 size = count;
135         }
136
137         if ((off & 1) && size) {
138                 u8 val;
139                 rio_read_config_8(dev, off, &val);
140                 data[off - init_off] = val;
141                 off++;
142                 size--;
143         }
144
145         if ((off & 3) && size > 2) {
146                 u16 val;
147                 rio_read_config_16(dev, off, &val);
148                 data[off - init_off] = (val >> 8) & 0xff;
149                 data[off - init_off + 1] = val & 0xff;
150                 off += 2;
151                 size -= 2;
152         }
153
154         while (size > 3) {
155                 u32 val;
156                 rio_read_config_32(dev, off, &val);
157                 data[off - init_off] = (val >> 24) & 0xff;
158                 data[off - init_off + 1] = (val >> 16) & 0xff;
159                 data[off - init_off + 2] = (val >> 8) & 0xff;
160                 data[off - init_off + 3] = val & 0xff;
161                 off += 4;
162                 size -= 4;
163         }
164
165         if (size >= 2) {
166                 u16 val;
167                 rio_read_config_16(dev, off, &val);
168                 data[off - init_off] = (val >> 8) & 0xff;
169                 data[off - init_off + 1] = val & 0xff;
170                 off += 2;
171                 size -= 2;
172         }
173
174         if (size > 0) {
175                 u8 val;
176                 rio_read_config_8(dev, off, &val);
177                 data[off - init_off] = val;
178                 off++;
179                 --size;
180         }
181
182         return count;
183 }
184
185 static ssize_t
186 rio_write_config(struct file *filp, struct kobject *kobj,
187                  struct bin_attribute *bin_attr,
188                  char *buf, loff_t off, size_t count)
189 {
190         struct rio_dev *dev =
191             to_rio_dev(container_of(kobj, struct device, kobj));
192         unsigned int size = count;
193         loff_t init_off = off;
194         u8 *data = (u8 *) buf;
195
196         if (off >= RIO_MAINT_SPACE_SZ)
197                 return 0;
198         if (off + count > RIO_MAINT_SPACE_SZ) {
199                 size = RIO_MAINT_SPACE_SZ - off;
200                 count = size;
201         }
202
203         if ((off & 1) && size) {
204                 rio_write_config_8(dev, off, data[off - init_off]);
205                 off++;
206                 size--;
207         }
208
209         if ((off & 3) && (size > 2)) {
210                 u16 val = data[off - init_off + 1];
211                 val |= (u16) data[off - init_off] << 8;
212                 rio_write_config_16(dev, off, val);
213                 off += 2;
214                 size -= 2;
215         }
216
217         while (size > 3) {
218                 u32 val = data[off - init_off + 3];
219                 val |= (u32) data[off - init_off + 2] << 8;
220                 val |= (u32) data[off - init_off + 1] << 16;
221                 val |= (u32) data[off - init_off] << 24;
222                 rio_write_config_32(dev, off, val);
223                 off += 4;
224                 size -= 4;
225         }
226
227         if (size >= 2) {
228                 u16 val = data[off - init_off + 1];
229                 val |= (u16) data[off - init_off] << 8;
230                 rio_write_config_16(dev, off, val);
231                 off += 2;
232                 size -= 2;
233         }
234
235         if (size) {
236                 rio_write_config_8(dev, off, data[off - init_off]);
237                 off++;
238                 --size;
239         }
240
241         return count;
242 }
243
244 static struct bin_attribute rio_config_attr = {
245         .attr = {
246                  .name = "config",
247                  .mode = S_IRUGO | S_IWUSR,
248                  },
249         .size = RIO_MAINT_SPACE_SZ,
250         .read = rio_read_config,
251         .write = rio_write_config,
252 };
253
254 /**
255  * rio_create_sysfs_dev_files - create RIO specific sysfs files
256  * @rdev: device whose entries should be created
257  *
258  * Create files when @rdev is added to sysfs.
259  */
260 int rio_create_sysfs_dev_files(struct rio_dev *rdev)
261 {
262         int err = 0;
263
264         err = device_create_bin_file(&rdev->dev, &rio_config_attr);
265
266         if (!err && (rdev->pef & RIO_PEF_SWITCH)) {
267                 err |= device_create_file(&rdev->dev, &dev_attr_routes);
268                 err |= device_create_file(&rdev->dev, &dev_attr_lnext);
269                 err |= device_create_file(&rdev->dev, &dev_attr_hopcount);
270         }
271
272         if (err)
273                 pr_warning("RIO: Failed to create attribute file(s) for %s\n",
274                            rio_name(rdev));
275
276         return err;
277 }
278
279 /**
280  * rio_remove_sysfs_dev_files - cleanup RIO specific sysfs files
281  * @rdev: device whose entries we should free
282  *
283  * Cleanup when @rdev is removed from sysfs.
284  */
285 void rio_remove_sysfs_dev_files(struct rio_dev *rdev)
286 {
287         device_remove_bin_file(&rdev->dev, &rio_config_attr);
288         if (rdev->pef & RIO_PEF_SWITCH) {
289                 device_remove_file(&rdev->dev, &dev_attr_routes);
290                 device_remove_file(&rdev->dev, &dev_attr_lnext);
291                 device_remove_file(&rdev->dev, &dev_attr_hopcount);
292         }
293 }
294
295 static ssize_t bus_scan_store(struct bus_type *bus, const char *buf,
296                                 size_t count)
297 {
298         long val;
299         int rc;
300
301         if (kstrtol(buf, 0, &val) < 0)
302                 return -EINVAL;
303
304         if (val == RIO_MPORT_ANY) {
305                 rc = rio_init_mports();
306                 goto exit;
307         }
308
309         if (val < 0 || val >= RIO_MAX_MPORTS)
310                 return -EINVAL;
311
312         rc = rio_mport_scan((int)val);
313 exit:
314         if (!rc)
315                 rc = count;
316
317         return rc;
318 }
319
320 struct bus_attribute rio_bus_attrs[] = {
321         __ATTR(scan, (S_IWUSR|S_IWGRP), NULL, bus_scan_store),
322         __ATTR_NULL
323 };