diff options
author | Inaky Perez-Gonzalez <inaky@linux.intel.com> | 2008-12-20 19:57:49 -0500 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@suse.de> | 2009-01-07 13:00:20 -0500 |
commit | c71228caf91ec6320b489dec5cd0087b64da9fb5 (patch) | |
tree | 947ccdbad749b6ae30ecdedbd557de34c068bb93 /drivers/net/wimax/i2400m/debugfs.c | |
parent | 3a35a1d0bdf7cc32cddc234b956605e6d4db4673 (diff) |
i2400m: debugfs controls
Expose knobs to control the device (induce reset, power saving,
querying tx or rx stats, internal debug information and debug level
manipulation).
Signed-off-by: Inaky Perez-Gonzalez <inaky@linux.intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Diffstat (limited to 'drivers/net/wimax/i2400m/debugfs.c')
-rw-r--r-- | drivers/net/wimax/i2400m/debugfs.c | 392 |
1 files changed, 392 insertions, 0 deletions
diff --git a/drivers/net/wimax/i2400m/debugfs.c b/drivers/net/wimax/i2400m/debugfs.c new file mode 100644 index 000000000000..626632985977 --- /dev/null +++ b/drivers/net/wimax/i2400m/debugfs.c | |||
@@ -0,0 +1,392 @@ | |||
1 | /* | ||
2 | * Intel Wireless WiMAX Connection 2400m | ||
3 | * Debugfs interfaces to manipulate driver and device information | ||
4 | * | ||
5 | * | ||
6 | * Copyright (C) 2007 Intel Corporation <linux-wimax@intel.com> | ||
7 | * Inaky Perez-Gonzalez <inaky.perez-gonzalez@intel.com> | ||
8 | * | ||
9 | * This program is free software; you can redistribute it and/or | ||
10 | * modify it under the terms of the GNU General Public License version | ||
11 | * 2 as published by the Free Software Foundation. | ||
12 | * | ||
13 | * This program is distributed in the hope that it will be useful, | ||
14 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
15 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
16 | * GNU General Public License for more details. | ||
17 | * | ||
18 | * You should have received a copy of the GNU General Public License | ||
19 | * along with this program; if not, write to the Free Software | ||
20 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA | ||
21 | * 02110-1301, USA. | ||
22 | */ | ||
23 | |||
24 | #include <linux/debugfs.h> | ||
25 | #include <linux/netdevice.h> | ||
26 | #include <linux/etherdevice.h> | ||
27 | #include <linux/spinlock.h> | ||
28 | #include <linux/device.h> | ||
29 | #include "i2400m.h" | ||
30 | |||
31 | |||
32 | #define D_SUBMODULE debugfs | ||
33 | #include "debug-levels.h" | ||
34 | |||
35 | static | ||
36 | int debugfs_netdev_queue_stopped_get(void *data, u64 *val) | ||
37 | { | ||
38 | struct i2400m *i2400m = data; | ||
39 | *val = netif_queue_stopped(i2400m->wimax_dev.net_dev); | ||
40 | return 0; | ||
41 | } | ||
42 | DEFINE_SIMPLE_ATTRIBUTE(fops_netdev_queue_stopped, | ||
43 | debugfs_netdev_queue_stopped_get, | ||
44 | NULL, "%llu\n"); | ||
45 | |||
46 | |||
47 | static | ||
48 | struct dentry *debugfs_create_netdev_queue_stopped( | ||
49 | const char *name, struct dentry *parent, struct i2400m *i2400m) | ||
50 | { | ||
51 | return debugfs_create_file(name, 0400, parent, i2400m, | ||
52 | &fops_netdev_queue_stopped); | ||
53 | } | ||
54 | |||
55 | |||
56 | /* | ||
57 | * inode->i_private has the @data argument to debugfs_create_file() | ||
58 | */ | ||
59 | static | ||
60 | int i2400m_stats_open(struct inode *inode, struct file *filp) | ||
61 | { | ||
62 | filp->private_data = inode->i_private; | ||
63 | return 0; | ||
64 | } | ||
65 | |||
66 | /* | ||
67 | * We don't allow partial reads of this file, as then the reader would | ||
68 | * get weirdly confused data as it is updated. | ||
69 | * | ||
70 | * So or you read it all or nothing; if you try to read with an offset | ||
71 | * != 0, we consider you are done reading. | ||
72 | */ | ||
73 | static | ||
74 | ssize_t i2400m_rx_stats_read(struct file *filp, char __user *buffer, | ||
75 | size_t count, loff_t *ppos) | ||
76 | { | ||
77 | struct i2400m *i2400m = filp->private_data; | ||
78 | char buf[128]; | ||
79 | unsigned long flags; | ||
80 | |||
81 | if (*ppos != 0) | ||
82 | return 0; | ||
83 | if (count < sizeof(buf)) | ||
84 | return -ENOSPC; | ||
85 | spin_lock_irqsave(&i2400m->rx_lock, flags); | ||
86 | snprintf(buf, sizeof(buf), "%u %u %u %u %u %u %u\n", | ||
87 | i2400m->rx_pl_num, i2400m->rx_pl_min, | ||
88 | i2400m->rx_pl_max, i2400m->rx_num, | ||
89 | i2400m->rx_size_acc, | ||
90 | i2400m->rx_size_min, i2400m->rx_size_max); | ||
91 | spin_unlock_irqrestore(&i2400m->rx_lock, flags); | ||
92 | return simple_read_from_buffer(buffer, count, ppos, buf, strlen(buf)); | ||
93 | } | ||
94 | |||
95 | |||
96 | /* Any write clears the stats */ | ||
97 | static | ||
98 | ssize_t i2400m_rx_stats_write(struct file *filp, const char __user *buffer, | ||
99 | size_t count, loff_t *ppos) | ||
100 | { | ||
101 | struct i2400m *i2400m = filp->private_data; | ||
102 | unsigned long flags; | ||
103 | |||
104 | spin_lock_irqsave(&i2400m->rx_lock, flags); | ||
105 | i2400m->rx_pl_num = 0; | ||
106 | i2400m->rx_pl_max = 0; | ||
107 | i2400m->rx_pl_min = UINT_MAX; | ||
108 | i2400m->rx_num = 0; | ||
109 | i2400m->rx_size_acc = 0; | ||
110 | i2400m->rx_size_min = UINT_MAX; | ||
111 | i2400m->rx_size_max = 0; | ||
112 | spin_unlock_irqrestore(&i2400m->rx_lock, flags); | ||
113 | return count; | ||
114 | } | ||
115 | |||
116 | static | ||
117 | const struct file_operations i2400m_rx_stats_fops = { | ||
118 | .owner = THIS_MODULE, | ||
119 | .open = i2400m_stats_open, | ||
120 | .read = i2400m_rx_stats_read, | ||
121 | .write = i2400m_rx_stats_write, | ||
122 | }; | ||
123 | |||
124 | |||
125 | /* See i2400m_rx_stats_read() */ | ||
126 | static | ||
127 | ssize_t i2400m_tx_stats_read(struct file *filp, char __user *buffer, | ||
128 | size_t count, loff_t *ppos) | ||
129 | { | ||
130 | struct i2400m *i2400m = filp->private_data; | ||
131 | char buf[128]; | ||
132 | unsigned long flags; | ||
133 | |||
134 | if (*ppos != 0) | ||
135 | return 0; | ||
136 | if (count < sizeof(buf)) | ||
137 | return -ENOSPC; | ||
138 | spin_lock_irqsave(&i2400m->tx_lock, flags); | ||
139 | snprintf(buf, sizeof(buf), "%u %u %u %u %u %u %u\n", | ||
140 | i2400m->tx_pl_num, i2400m->tx_pl_min, | ||
141 | i2400m->tx_pl_max, i2400m->tx_num, | ||
142 | i2400m->tx_size_acc, | ||
143 | i2400m->tx_size_min, i2400m->tx_size_max); | ||
144 | spin_unlock_irqrestore(&i2400m->tx_lock, flags); | ||
145 | return simple_read_from_buffer(buffer, count, ppos, buf, strlen(buf)); | ||
146 | } | ||
147 | |||
148 | /* Any write clears the stats */ | ||
149 | static | ||
150 | ssize_t i2400m_tx_stats_write(struct file *filp, const char __user *buffer, | ||
151 | size_t count, loff_t *ppos) | ||
152 | { | ||
153 | struct i2400m *i2400m = filp->private_data; | ||
154 | unsigned long flags; | ||
155 | |||
156 | spin_lock_irqsave(&i2400m->tx_lock, flags); | ||
157 | i2400m->tx_pl_num = 0; | ||
158 | i2400m->tx_pl_max = 0; | ||
159 | i2400m->tx_pl_min = UINT_MAX; | ||
160 | i2400m->tx_num = 0; | ||
161 | i2400m->tx_size_acc = 0; | ||
162 | i2400m->tx_size_min = UINT_MAX; | ||
163 | i2400m->tx_size_max = 0; | ||
164 | spin_unlock_irqrestore(&i2400m->tx_lock, flags); | ||
165 | return count; | ||
166 | } | ||
167 | |||
168 | static | ||
169 | const struct file_operations i2400m_tx_stats_fops = { | ||
170 | .owner = THIS_MODULE, | ||
171 | .open = i2400m_stats_open, | ||
172 | .read = i2400m_tx_stats_read, | ||
173 | .write = i2400m_tx_stats_write, | ||
174 | }; | ||
175 | |||
176 | |||
177 | /* Write 1 to ask the device to go into suspend */ | ||
178 | static | ||
179 | int debugfs_i2400m_suspend_set(void *data, u64 val) | ||
180 | { | ||
181 | int result; | ||
182 | struct i2400m *i2400m = data; | ||
183 | result = i2400m_cmd_enter_powersave(i2400m); | ||
184 | if (result >= 0) | ||
185 | result = 0; | ||
186 | return result; | ||
187 | } | ||
188 | DEFINE_SIMPLE_ATTRIBUTE(fops_i2400m_suspend, | ||
189 | NULL, debugfs_i2400m_suspend_set, | ||
190 | "%llu\n"); | ||
191 | |||
192 | static | ||
193 | struct dentry *debugfs_create_i2400m_suspend( | ||
194 | const char *name, struct dentry *parent, struct i2400m *i2400m) | ||
195 | { | ||
196 | return debugfs_create_file(name, 0200, parent, i2400m, | ||
197 | &fops_i2400m_suspend); | ||
198 | } | ||
199 | |||
200 | |||
201 | /* | ||
202 | * Reset the device | ||
203 | * | ||
204 | * Write 0 to ask the device to soft reset, 1 to cold reset, 2 to bus | ||
205 | * reset (as defined by enum i2400m_reset_type). | ||
206 | */ | ||
207 | static | ||
208 | int debugfs_i2400m_reset_set(void *data, u64 val) | ||
209 | { | ||
210 | int result; | ||
211 | struct i2400m *i2400m = data; | ||
212 | enum i2400m_reset_type rt = val; | ||
213 | switch(rt) { | ||
214 | case I2400M_RT_WARM: | ||
215 | case I2400M_RT_COLD: | ||
216 | case I2400M_RT_BUS: | ||
217 | result = i2400m->bus_reset(i2400m, rt); | ||
218 | if (result >= 0) | ||
219 | result = 0; | ||
220 | default: | ||
221 | result = -EINVAL; | ||
222 | } | ||
223 | return result; | ||
224 | } | ||
225 | DEFINE_SIMPLE_ATTRIBUTE(fops_i2400m_reset, | ||
226 | NULL, debugfs_i2400m_reset_set, | ||
227 | "%llu\n"); | ||
228 | |||
229 | static | ||
230 | struct dentry *debugfs_create_i2400m_reset( | ||
231 | const char *name, struct dentry *parent, struct i2400m *i2400m) | ||
232 | { | ||
233 | return debugfs_create_file(name, 0200, parent, i2400m, | ||
234 | &fops_i2400m_reset); | ||
235 | } | ||
236 | |||
237 | /* | ||
238 | * Debug levels control; see debug.h | ||
239 | */ | ||
240 | struct d_level D_LEVEL[] = { | ||
241 | D_SUBMODULE_DEFINE(control), | ||
242 | D_SUBMODULE_DEFINE(driver), | ||
243 | D_SUBMODULE_DEFINE(debugfs), | ||
244 | D_SUBMODULE_DEFINE(fw), | ||
245 | D_SUBMODULE_DEFINE(netdev), | ||
246 | D_SUBMODULE_DEFINE(rfkill), | ||
247 | D_SUBMODULE_DEFINE(rx), | ||
248 | D_SUBMODULE_DEFINE(tx), | ||
249 | }; | ||
250 | size_t D_LEVEL_SIZE = ARRAY_SIZE(D_LEVEL); | ||
251 | |||
252 | #define __debugfs_register(prefix, name, parent) \ | ||
253 | do { \ | ||
254 | result = d_level_register_debugfs(prefix, name, parent); \ | ||
255 | if (result < 0) \ | ||
256 | goto error; \ | ||
257 | } while (0) | ||
258 | |||
259 | |||
260 | int i2400m_debugfs_add(struct i2400m *i2400m) | ||
261 | { | ||
262 | int result; | ||
263 | struct device *dev = i2400m_dev(i2400m); | ||
264 | struct dentry *dentry = i2400m->wimax_dev.debugfs_dentry; | ||
265 | struct dentry *fd; | ||
266 | |||
267 | dentry = debugfs_create_dir("i2400m", dentry); | ||
268 | result = PTR_ERR(dentry); | ||
269 | if (IS_ERR(dentry)) { | ||
270 | if (result == -ENODEV) | ||
271 | result = 0; /* No debugfs support */ | ||
272 | goto error; | ||
273 | } | ||
274 | i2400m->debugfs_dentry = dentry; | ||
275 | __debugfs_register("dl_", control, dentry); | ||
276 | __debugfs_register("dl_", driver, dentry); | ||
277 | __debugfs_register("dl_", debugfs, dentry); | ||
278 | __debugfs_register("dl_", fw, dentry); | ||
279 | __debugfs_register("dl_", netdev, dentry); | ||
280 | __debugfs_register("dl_", rfkill, dentry); | ||
281 | __debugfs_register("dl_", rx, dentry); | ||
282 | __debugfs_register("dl_", tx, dentry); | ||
283 | |||
284 | fd = debugfs_create_size_t("tx_in", 0400, dentry, | ||
285 | &i2400m->tx_in); | ||
286 | result = PTR_ERR(fd); | ||
287 | if (IS_ERR(fd) && result != -ENODEV) { | ||
288 | dev_err(dev, "Can't create debugfs entry " | ||
289 | "tx_in: %d\n", result); | ||
290 | goto error; | ||
291 | } | ||
292 | |||
293 | fd = debugfs_create_size_t("tx_out", 0400, dentry, | ||
294 | &i2400m->tx_out); | ||
295 | result = PTR_ERR(fd); | ||
296 | if (IS_ERR(fd) && result != -ENODEV) { | ||
297 | dev_err(dev, "Can't create debugfs entry " | ||
298 | "tx_out: %d\n", result); | ||
299 | goto error; | ||
300 | } | ||
301 | |||
302 | fd = debugfs_create_u32("state", 0600, dentry, | ||
303 | &i2400m->state); | ||
304 | result = PTR_ERR(fd); | ||
305 | if (IS_ERR(fd) && result != -ENODEV) { | ||
306 | dev_err(dev, "Can't create debugfs entry " | ||
307 | "state: %d\n", result); | ||
308 | goto error; | ||
309 | } | ||
310 | |||
311 | /* | ||
312 | * Trace received messages from user space | ||
313 | * | ||
314 | * In order to tap the bidirectional message stream in the | ||
315 | * 'msg' pipe, user space can read from the 'msg' pipe; | ||
316 | * however, due to limitations in libnl, we can't know what | ||
317 | * the different applications are sending down to the kernel. | ||
318 | * | ||
319 | * So we have this hack where the driver will echo any message | ||
320 | * received on the msg pipe from user space [through a call to | ||
321 | * wimax_dev->op_msg_from_user() into | ||
322 | * i2400m_op_msg_from_user()] into the 'trace' pipe that this | ||
323 | * driver creates. | ||
324 | * | ||
325 | * So then, reading from both the 'trace' and 'msg' pipes in | ||
326 | * user space will provide a full dump of the traffic. | ||
327 | * | ||
328 | * Write 1 to activate, 0 to clear. | ||
329 | * | ||
330 | * It is not really very atomic, but it is also not too | ||
331 | * critical. | ||
332 | */ | ||
333 | fd = debugfs_create_u8("trace_msg_from_user", 0600, dentry, | ||
334 | &i2400m->trace_msg_from_user); | ||
335 | result = PTR_ERR(fd); | ||
336 | if (IS_ERR(fd) && result != -ENODEV) { | ||
337 | dev_err(dev, "Can't create debugfs entry " | ||
338 | "trace_msg_from_user: %d\n", result); | ||
339 | goto error; | ||
340 | } | ||
341 | |||
342 | fd = debugfs_create_netdev_queue_stopped("netdev_queue_stopped", | ||
343 | dentry, i2400m); | ||
344 | result = PTR_ERR(fd); | ||
345 | if (IS_ERR(fd) && result != -ENODEV) { | ||
346 | dev_err(dev, "Can't create debugfs entry " | ||
347 | "netdev_queue_stopped: %d\n", result); | ||
348 | goto error; | ||
349 | } | ||
350 | |||
351 | fd = debugfs_create_file("rx_stats", 0600, dentry, i2400m, | ||
352 | &i2400m_rx_stats_fops); | ||
353 | result = PTR_ERR(fd); | ||
354 | if (IS_ERR(fd) && result != -ENODEV) { | ||
355 | dev_err(dev, "Can't create debugfs entry " | ||
356 | "rx_stats: %d\n", result); | ||
357 | goto error; | ||
358 | } | ||
359 | |||
360 | fd = debugfs_create_file("tx_stats", 0600, dentry, i2400m, | ||
361 | &i2400m_tx_stats_fops); | ||
362 | result = PTR_ERR(fd); | ||
363 | if (IS_ERR(fd) && result != -ENODEV) { | ||
364 | dev_err(dev, "Can't create debugfs entry " | ||
365 | "tx_stats: %d\n", result); | ||
366 | goto error; | ||
367 | } | ||
368 | |||
369 | fd = debugfs_create_i2400m_suspend("suspend", dentry, i2400m); | ||
370 | result = PTR_ERR(fd); | ||
371 | if (IS_ERR(fd) && result != -ENODEV) { | ||
372 | dev_err(dev, "Can't create debugfs entry suspend: %d\n", | ||
373 | result); | ||
374 | goto error; | ||
375 | } | ||
376 | |||
377 | fd = debugfs_create_i2400m_reset("reset", dentry, i2400m); | ||
378 | result = PTR_ERR(fd); | ||
379 | if (IS_ERR(fd) && result != -ENODEV) { | ||
380 | dev_err(dev, "Can't create debugfs entry reset: %d\n", result); | ||
381 | goto error; | ||
382 | } | ||
383 | |||
384 | result = 0; | ||
385 | error: | ||
386 | return result; | ||
387 | } | ||
388 | |||
389 | void i2400m_debugfs_rm(struct i2400m *i2400m) | ||
390 | { | ||
391 | debugfs_remove_recursive(i2400m->debugfs_dentry); | ||
392 | } | ||