0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013 #include <linux/fs.h>
0014 #include <linux/debugfs.h>
0015 #include <linux/slab.h>
0016 #include <linux/netdevice.h>
0017 #include <linux/pci.h>
0018 #include <linux/mutex.h>
0019
0020 #include "b43.h"
0021 #include "main.h"
0022 #include "debugfs.h"
0023 #include "dma.h"
0024 #include "xmit.h"
0025
0026
0027
0028 static struct dentry *rootdir;
0029
0030 struct b43_debugfs_fops {
0031 ssize_t (*read)(struct b43_wldev *dev, char *buf, size_t bufsize);
0032 int (*write)(struct b43_wldev *dev, const char *buf, size_t count);
0033 struct file_operations fops;
0034
0035 size_t file_struct_offset;
0036 };
0037
0038 static inline
0039 struct b43_dfs_file *fops_to_dfs_file(struct b43_wldev *dev,
0040 const struct b43_debugfs_fops *dfops)
0041 {
0042 void *p;
0043
0044 p = dev->dfsentry;
0045 p += dfops->file_struct_offset;
0046
0047 return p;
0048 }
0049
0050
0051 #define fappend(fmt, x...) \
0052 do { \
0053 if (bufsize - count) \
0054 count += scnprintf(buf + count, \
0055 bufsize - count, \
0056 fmt , ##x); \
0057 else \
0058 printk(KERN_ERR "b43: fappend overflow\n"); \
0059 } while (0)
0060
0061
0062
0063 #define B43_MAX_SHM_ROUTING 4
0064 #define B43_MAX_SHM_ADDR 0xFFFF
0065
0066 static ssize_t shm16read__read_file(struct b43_wldev *dev,
0067 char *buf, size_t bufsize)
0068 {
0069 ssize_t count = 0;
0070 unsigned int routing, addr;
0071 u16 val;
0072
0073 routing = dev->dfsentry->shm16read_routing_next;
0074 addr = dev->dfsentry->shm16read_addr_next;
0075 if ((routing > B43_MAX_SHM_ROUTING) ||
0076 (addr > B43_MAX_SHM_ADDR))
0077 return -EDESTADDRREQ;
0078
0079 val = b43_shm_read16(dev, routing, addr);
0080 fappend("0x%04X\n", val);
0081
0082 return count;
0083 }
0084
0085 static int shm16read__write_file(struct b43_wldev *dev,
0086 const char *buf, size_t count)
0087 {
0088 unsigned int routing, addr;
0089 int res;
0090
0091 res = sscanf(buf, "0x%X 0x%X", &routing, &addr);
0092 if (res != 2)
0093 return -EINVAL;
0094 if (routing > B43_MAX_SHM_ROUTING)
0095 return -EADDRNOTAVAIL;
0096 if (addr > B43_MAX_SHM_ADDR)
0097 return -EADDRNOTAVAIL;
0098 if (routing == B43_SHM_SHARED) {
0099 if ((addr % 2) != 0)
0100 return -EADDRNOTAVAIL;
0101 }
0102
0103 dev->dfsentry->shm16read_routing_next = routing;
0104 dev->dfsentry->shm16read_addr_next = addr;
0105
0106 return 0;
0107 }
0108
0109 static int shm16write__write_file(struct b43_wldev *dev,
0110 const char *buf, size_t count)
0111 {
0112 unsigned int routing, addr, mask, set;
0113 u16 val;
0114 int res;
0115
0116 res = sscanf(buf, "0x%X 0x%X 0x%X 0x%X",
0117 &routing, &addr, &mask, &set);
0118 if (res != 4)
0119 return -EINVAL;
0120 if (routing > B43_MAX_SHM_ROUTING)
0121 return -EADDRNOTAVAIL;
0122 if (addr > B43_MAX_SHM_ADDR)
0123 return -EADDRNOTAVAIL;
0124 if (routing == B43_SHM_SHARED) {
0125 if ((addr % 2) != 0)
0126 return -EADDRNOTAVAIL;
0127 }
0128 if ((mask > 0xFFFF) || (set > 0xFFFF))
0129 return -E2BIG;
0130
0131 if (mask == 0)
0132 val = 0;
0133 else
0134 val = b43_shm_read16(dev, routing, addr);
0135 val &= mask;
0136 val |= set;
0137 b43_shm_write16(dev, routing, addr, val);
0138
0139 return 0;
0140 }
0141
0142 static ssize_t shm32read__read_file(struct b43_wldev *dev,
0143 char *buf, size_t bufsize)
0144 {
0145 ssize_t count = 0;
0146 unsigned int routing, addr;
0147 u32 val;
0148
0149 routing = dev->dfsentry->shm32read_routing_next;
0150 addr = dev->dfsentry->shm32read_addr_next;
0151 if ((routing > B43_MAX_SHM_ROUTING) ||
0152 (addr > B43_MAX_SHM_ADDR))
0153 return -EDESTADDRREQ;
0154
0155 val = b43_shm_read32(dev, routing, addr);
0156 fappend("0x%08X\n", val);
0157
0158 return count;
0159 }
0160
0161 static int shm32read__write_file(struct b43_wldev *dev,
0162 const char *buf, size_t count)
0163 {
0164 unsigned int routing, addr;
0165 int res;
0166
0167 res = sscanf(buf, "0x%X 0x%X", &routing, &addr);
0168 if (res != 2)
0169 return -EINVAL;
0170 if (routing > B43_MAX_SHM_ROUTING)
0171 return -EADDRNOTAVAIL;
0172 if (addr > B43_MAX_SHM_ADDR)
0173 return -EADDRNOTAVAIL;
0174 if (routing == B43_SHM_SHARED) {
0175 if ((addr % 2) != 0)
0176 return -EADDRNOTAVAIL;
0177 }
0178
0179 dev->dfsentry->shm32read_routing_next = routing;
0180 dev->dfsentry->shm32read_addr_next = addr;
0181
0182 return 0;
0183 }
0184
0185 static int shm32write__write_file(struct b43_wldev *dev,
0186 const char *buf, size_t count)
0187 {
0188 unsigned int routing, addr, mask, set;
0189 u32 val;
0190 int res;
0191
0192 res = sscanf(buf, "0x%X 0x%X 0x%X 0x%X",
0193 &routing, &addr, &mask, &set);
0194 if (res != 4)
0195 return -EINVAL;
0196 if (routing > B43_MAX_SHM_ROUTING)
0197 return -EADDRNOTAVAIL;
0198 if (addr > B43_MAX_SHM_ADDR)
0199 return -EADDRNOTAVAIL;
0200 if (routing == B43_SHM_SHARED) {
0201 if ((addr % 2) != 0)
0202 return -EADDRNOTAVAIL;
0203 }
0204 if ((mask > 0xFFFFFFFF) || (set > 0xFFFFFFFF))
0205 return -E2BIG;
0206
0207 if (mask == 0)
0208 val = 0;
0209 else
0210 val = b43_shm_read32(dev, routing, addr);
0211 val &= mask;
0212 val |= set;
0213 b43_shm_write32(dev, routing, addr, val);
0214
0215 return 0;
0216 }
0217
0218
0219 #define B43_MAX_MMIO_ACCESS (0xF00 - 1)
0220
0221 static ssize_t mmio16read__read_file(struct b43_wldev *dev,
0222 char *buf, size_t bufsize)
0223 {
0224 ssize_t count = 0;
0225 unsigned int addr;
0226 u16 val;
0227
0228 addr = dev->dfsentry->mmio16read_next;
0229 if (addr > B43_MAX_MMIO_ACCESS)
0230 return -EDESTADDRREQ;
0231
0232 val = b43_read16(dev, addr);
0233 fappend("0x%04X\n", val);
0234
0235 return count;
0236 }
0237
0238 static int mmio16read__write_file(struct b43_wldev *dev,
0239 const char *buf, size_t count)
0240 {
0241 unsigned int addr;
0242 int res;
0243
0244 res = sscanf(buf, "0x%X", &addr);
0245 if (res != 1)
0246 return -EINVAL;
0247 if (addr > B43_MAX_MMIO_ACCESS)
0248 return -EADDRNOTAVAIL;
0249 if ((addr % 2) != 0)
0250 return -EINVAL;
0251
0252 dev->dfsentry->mmio16read_next = addr;
0253
0254 return 0;
0255 }
0256
0257 static int mmio16write__write_file(struct b43_wldev *dev,
0258 const char *buf, size_t count)
0259 {
0260 unsigned int addr, mask, set;
0261 int res;
0262 u16 val;
0263
0264 res = sscanf(buf, "0x%X 0x%X 0x%X", &addr, &mask, &set);
0265 if (res != 3)
0266 return -EINVAL;
0267 if (addr > B43_MAX_MMIO_ACCESS)
0268 return -EADDRNOTAVAIL;
0269 if ((mask > 0xFFFF) || (set > 0xFFFF))
0270 return -E2BIG;
0271 if ((addr % 2) != 0)
0272 return -EINVAL;
0273
0274 if (mask == 0)
0275 val = 0;
0276 else
0277 val = b43_read16(dev, addr);
0278 val &= mask;
0279 val |= set;
0280 b43_write16(dev, addr, val);
0281
0282 return 0;
0283 }
0284
0285 static ssize_t mmio32read__read_file(struct b43_wldev *dev,
0286 char *buf, size_t bufsize)
0287 {
0288 ssize_t count = 0;
0289 unsigned int addr;
0290 u32 val;
0291
0292 addr = dev->dfsentry->mmio32read_next;
0293 if (addr > B43_MAX_MMIO_ACCESS)
0294 return -EDESTADDRREQ;
0295
0296 val = b43_read32(dev, addr);
0297 fappend("0x%08X\n", val);
0298
0299 return count;
0300 }
0301
0302 static int mmio32read__write_file(struct b43_wldev *dev,
0303 const char *buf, size_t count)
0304 {
0305 unsigned int addr;
0306 int res;
0307
0308 res = sscanf(buf, "0x%X", &addr);
0309 if (res != 1)
0310 return -EINVAL;
0311 if (addr > B43_MAX_MMIO_ACCESS)
0312 return -EADDRNOTAVAIL;
0313 if ((addr % 4) != 0)
0314 return -EINVAL;
0315
0316 dev->dfsentry->mmio32read_next = addr;
0317
0318 return 0;
0319 }
0320
0321 static int mmio32write__write_file(struct b43_wldev *dev,
0322 const char *buf, size_t count)
0323 {
0324 unsigned int addr, mask, set;
0325 int res;
0326 u32 val;
0327
0328 res = sscanf(buf, "0x%X 0x%X 0x%X", &addr, &mask, &set);
0329 if (res != 3)
0330 return -EINVAL;
0331 if (addr > B43_MAX_MMIO_ACCESS)
0332 return -EADDRNOTAVAIL;
0333 if ((mask > 0xFFFFFFFF) || (set > 0xFFFFFFFF))
0334 return -E2BIG;
0335 if ((addr % 4) != 0)
0336 return -EINVAL;
0337
0338 if (mask == 0)
0339 val = 0;
0340 else
0341 val = b43_read32(dev, addr);
0342 val &= mask;
0343 val |= set;
0344 b43_write32(dev, addr, val);
0345
0346 return 0;
0347 }
0348
0349 static ssize_t txstat_read_file(struct b43_wldev *dev,
0350 char *buf, size_t bufsize)
0351 {
0352 struct b43_txstatus_log *log = &dev->dfsentry->txstatlog;
0353 ssize_t count = 0;
0354 int i, idx;
0355 struct b43_txstatus *stat;
0356
0357 if (log->end < 0) {
0358 fappend("Nothing transmitted, yet\n");
0359 goto out;
0360 }
0361 fappend("b43 TX status reports:\n\n"
0362 "index | cookie | seq | phy_stat | frame_count | "
0363 "rts_count | supp_reason | pm_indicated | "
0364 "intermediate | for_ampdu | acked\n" "---\n");
0365 i = log->end + 1;
0366 idx = 0;
0367 while (1) {
0368 if (i == B43_NR_LOGGED_TXSTATUS)
0369 i = 0;
0370 stat = &(log->log[i]);
0371 if (stat->cookie) {
0372 fappend("%03d | "
0373 "0x%04X | 0x%04X | 0x%02X | "
0374 "0x%X | 0x%X | "
0375 "%u | %u | "
0376 "%u | %u | %u\n",
0377 idx,
0378 stat->cookie, stat->seq, stat->phy_stat,
0379 stat->frame_count, stat->rts_count,
0380 stat->supp_reason, stat->pm_indicated,
0381 stat->intermediate, stat->for_ampdu,
0382 stat->acked);
0383 idx++;
0384 }
0385 if (i == log->end)
0386 break;
0387 i++;
0388 }
0389 out:
0390
0391 return count;
0392 }
0393
0394 static int restart_write_file(struct b43_wldev *dev,
0395 const char *buf, size_t count)
0396 {
0397 int err = 0;
0398
0399 if (count > 0 && buf[0] == '1') {
0400 b43_controller_restart(dev, "manually restarted");
0401 } else
0402 err = -EINVAL;
0403
0404 return err;
0405 }
0406
0407 static unsigned long calc_expire_secs(unsigned long now,
0408 unsigned long time,
0409 unsigned long expire)
0410 {
0411 expire = time + expire;
0412
0413 if (time_after(now, expire))
0414 return 0;
0415 if (expire < now) {
0416
0417 expire -= MAX_JIFFY_OFFSET;
0418 now -= MAX_JIFFY_OFFSET;
0419 }
0420 B43_WARN_ON(expire < now);
0421
0422 return (expire - now) / HZ;
0423 }
0424
0425 static ssize_t loctls_read_file(struct b43_wldev *dev,
0426 char *buf, size_t bufsize)
0427 {
0428 ssize_t count = 0;
0429 struct b43_txpower_lo_control *lo;
0430 int i, err = 0;
0431 struct b43_lo_calib *cal;
0432 unsigned long now = jiffies;
0433 struct b43_phy *phy = &dev->phy;
0434
0435 if (phy->type != B43_PHYTYPE_G) {
0436 fappend("Device is not a G-PHY\n");
0437 err = -ENODEV;
0438 goto out;
0439 }
0440 lo = phy->g->lo_control;
0441 fappend("-- Local Oscillator calibration data --\n\n");
0442 fappend("HW-power-control enabled: %d\n",
0443 dev->phy.hardware_power_control);
0444 fappend("TX Bias: 0x%02X, TX Magn: 0x%02X (expire in %lu sec)\n",
0445 lo->tx_bias, lo->tx_magn,
0446 calc_expire_secs(now, lo->txctl_measured_time,
0447 B43_LO_TXCTL_EXPIRE));
0448 fappend("Power Vector: 0x%08X%08X (expires in %lu sec)\n",
0449 (unsigned int)((lo->power_vector & 0xFFFFFFFF00000000ULL) >> 32),
0450 (unsigned int)(lo->power_vector & 0x00000000FFFFFFFFULL),
0451 calc_expire_secs(now, lo->pwr_vec_read_time,
0452 B43_LO_PWRVEC_EXPIRE));
0453
0454 fappend("\nCalibrated settings:\n");
0455 list_for_each_entry(cal, &lo->calib_list, list) {
0456 bool active;
0457
0458 active = (b43_compare_bbatt(&cal->bbatt, &phy->g->bbatt) &&
0459 b43_compare_rfatt(&cal->rfatt, &phy->g->rfatt));
0460 fappend("BB(%d), RF(%d,%d) -> I=%d, Q=%d "
0461 "(expires in %lu sec)%s\n",
0462 cal->bbatt.att,
0463 cal->rfatt.att, cal->rfatt.with_padmix,
0464 cal->ctl.i, cal->ctl.q,
0465 calc_expire_secs(now, cal->calib_time,
0466 B43_LO_CALIB_EXPIRE),
0467 active ? " ACTIVE" : "");
0468 }
0469
0470 fappend("\nUsed RF attenuation values: Value(WithPadmix flag)\n");
0471 for (i = 0; i < lo->rfatt_list.len; i++) {
0472 fappend("%u(%d), ",
0473 lo->rfatt_list.list[i].att,
0474 lo->rfatt_list.list[i].with_padmix);
0475 }
0476 fappend("\n");
0477 fappend("\nUsed Baseband attenuation values:\n");
0478 for (i = 0; i < lo->bbatt_list.len; i++) {
0479 fappend("%u, ",
0480 lo->bbatt_list.list[i].att);
0481 }
0482 fappend("\n");
0483
0484 out:
0485 return err ? err : count;
0486 }
0487
0488 #undef fappend
0489
0490 static ssize_t b43_debugfs_read(struct file *file, char __user *userbuf,
0491 size_t count, loff_t *ppos)
0492 {
0493 struct b43_wldev *dev;
0494 struct b43_debugfs_fops *dfops;
0495 struct b43_dfs_file *dfile;
0496 ssize_t ret;
0497 char *buf;
0498 const size_t bufsize = 1024 * 16;
0499 const size_t buforder = get_order(bufsize);
0500 int err = 0;
0501
0502 if (!count)
0503 return 0;
0504 dev = file->private_data;
0505 if (!dev)
0506 return -ENODEV;
0507
0508 mutex_lock(&dev->wl->mutex);
0509 if (b43_status(dev) < B43_STAT_INITIALIZED) {
0510 err = -ENODEV;
0511 goto out_unlock;
0512 }
0513
0514 dfops = container_of(debugfs_real_fops(file),
0515 struct b43_debugfs_fops, fops);
0516 if (!dfops->read) {
0517 err = -ENOSYS;
0518 goto out_unlock;
0519 }
0520 dfile = fops_to_dfs_file(dev, dfops);
0521
0522 if (!dfile->buffer) {
0523 buf = (char *)__get_free_pages(GFP_KERNEL, buforder);
0524 if (!buf) {
0525 err = -ENOMEM;
0526 goto out_unlock;
0527 }
0528 memset(buf, 0, bufsize);
0529 ret = dfops->read(dev, buf, bufsize);
0530 if (ret <= 0) {
0531 free_pages((unsigned long)buf, buforder);
0532 err = ret;
0533 goto out_unlock;
0534 }
0535 dfile->data_len = ret;
0536 dfile->buffer = buf;
0537 }
0538
0539 ret = simple_read_from_buffer(userbuf, count, ppos,
0540 dfile->buffer,
0541 dfile->data_len);
0542 if (*ppos >= dfile->data_len) {
0543 free_pages((unsigned long)dfile->buffer, buforder);
0544 dfile->buffer = NULL;
0545 dfile->data_len = 0;
0546 }
0547 out_unlock:
0548 mutex_unlock(&dev->wl->mutex);
0549
0550 return err ? err : ret;
0551 }
0552
0553 static ssize_t b43_debugfs_write(struct file *file,
0554 const char __user *userbuf,
0555 size_t count, loff_t *ppos)
0556 {
0557 struct b43_wldev *dev;
0558 struct b43_debugfs_fops *dfops;
0559 char *buf;
0560 int err = 0;
0561
0562 if (!count)
0563 return 0;
0564 if (count > PAGE_SIZE)
0565 return -E2BIG;
0566 dev = file->private_data;
0567 if (!dev)
0568 return -ENODEV;
0569
0570 mutex_lock(&dev->wl->mutex);
0571 if (b43_status(dev) < B43_STAT_INITIALIZED) {
0572 err = -ENODEV;
0573 goto out_unlock;
0574 }
0575
0576 dfops = container_of(debugfs_real_fops(file),
0577 struct b43_debugfs_fops, fops);
0578 if (!dfops->write) {
0579 err = -ENOSYS;
0580 goto out_unlock;
0581 }
0582
0583 buf = (char *)get_zeroed_page(GFP_KERNEL);
0584 if (!buf) {
0585 err = -ENOMEM;
0586 goto out_unlock;
0587 }
0588 if (copy_from_user(buf, userbuf, count)) {
0589 err = -EFAULT;
0590 goto out_freepage;
0591 }
0592 err = dfops->write(dev, buf, count);
0593 if (err)
0594 goto out_freepage;
0595
0596 out_freepage:
0597 free_page((unsigned long)buf);
0598 out_unlock:
0599 mutex_unlock(&dev->wl->mutex);
0600
0601 return err ? err : count;
0602 }
0603
0604
0605 #define B43_DEBUGFS_FOPS(name, _read, _write) \
0606 static struct b43_debugfs_fops fops_##name = { \
0607 .read = _read, \
0608 .write = _write, \
0609 .fops = { \
0610 .open = simple_open, \
0611 .read = b43_debugfs_read, \
0612 .write = b43_debugfs_write, \
0613 .llseek = generic_file_llseek, \
0614 }, \
0615 .file_struct_offset = offsetof(struct b43_dfsentry, \
0616 file_##name), \
0617 }
0618
0619 B43_DEBUGFS_FOPS(shm16read, shm16read__read_file, shm16read__write_file);
0620 B43_DEBUGFS_FOPS(shm16write, NULL, shm16write__write_file);
0621 B43_DEBUGFS_FOPS(shm32read, shm32read__read_file, shm32read__write_file);
0622 B43_DEBUGFS_FOPS(shm32write, NULL, shm32write__write_file);
0623 B43_DEBUGFS_FOPS(mmio16read, mmio16read__read_file, mmio16read__write_file);
0624 B43_DEBUGFS_FOPS(mmio16write, NULL, mmio16write__write_file);
0625 B43_DEBUGFS_FOPS(mmio32read, mmio32read__read_file, mmio32read__write_file);
0626 B43_DEBUGFS_FOPS(mmio32write, NULL, mmio32write__write_file);
0627 B43_DEBUGFS_FOPS(txstat, txstat_read_file, NULL);
0628 B43_DEBUGFS_FOPS(restart, NULL, restart_write_file);
0629 B43_DEBUGFS_FOPS(loctls, loctls_read_file, NULL);
0630
0631
0632 bool b43_debug(struct b43_wldev *dev, enum b43_dyndbg feature)
0633 {
0634 bool enabled;
0635
0636 enabled = (dev->dfsentry && dev->dfsentry->dyn_debug[feature]);
0637 if (unlikely(enabled)) {
0638
0639
0640 b43_modparam_verbose = B43_VERBOSITY_MAX;
0641 }
0642
0643 return enabled;
0644 }
0645
0646 static void b43_add_dynamic_debug(struct b43_wldev *dev)
0647 {
0648 struct b43_dfsentry *e = dev->dfsentry;
0649
0650 #define add_dyn_dbg(name, id, initstate) do { \
0651 e->dyn_debug[id] = (initstate); \
0652 debugfs_create_bool(name, 0600, e->subdir, \
0653 &(e->dyn_debug[id])); \
0654 } while (0)
0655
0656 add_dyn_dbg("debug_xmitpower", B43_DBG_XMITPOWER, false);
0657 add_dyn_dbg("debug_dmaoverflow", B43_DBG_DMAOVERFLOW, false);
0658 add_dyn_dbg("debug_dmaverbose", B43_DBG_DMAVERBOSE, false);
0659 add_dyn_dbg("debug_pwork_fast", B43_DBG_PWORK_FAST, false);
0660 add_dyn_dbg("debug_pwork_stop", B43_DBG_PWORK_STOP, false);
0661 add_dyn_dbg("debug_lo", B43_DBG_LO, false);
0662 add_dyn_dbg("debug_firmware", B43_DBG_FIRMWARE, false);
0663 add_dyn_dbg("debug_keys", B43_DBG_KEYS, false);
0664 add_dyn_dbg("debug_verbose_stats", B43_DBG_VERBOSESTATS, false);
0665
0666 #undef add_dyn_dbg
0667 }
0668
0669 void b43_debugfs_add_device(struct b43_wldev *dev)
0670 {
0671 struct b43_dfsentry *e;
0672 struct b43_txstatus_log *log;
0673 char devdir[16];
0674
0675 B43_WARN_ON(!dev);
0676 e = kzalloc(sizeof(*e), GFP_KERNEL);
0677 if (!e) {
0678 b43err(dev->wl, "debugfs: add device OOM\n");
0679 return;
0680 }
0681 e->dev = dev;
0682 log = &e->txstatlog;
0683 log->log = kcalloc(B43_NR_LOGGED_TXSTATUS,
0684 sizeof(struct b43_txstatus), GFP_KERNEL);
0685 if (!log->log) {
0686 b43err(dev->wl, "debugfs: add device txstatus OOM\n");
0687 kfree(e);
0688 return;
0689 }
0690 log->end = -1;
0691
0692 dev->dfsentry = e;
0693
0694 snprintf(devdir, sizeof(devdir), "%s", wiphy_name(dev->wl->hw->wiphy));
0695 e->subdir = debugfs_create_dir(devdir, rootdir);
0696
0697 e->mmio16read_next = 0xFFFF;
0698 e->mmio32read_next = 0xFFFF;
0699 e->shm16read_routing_next = 0xFFFFFFFF;
0700 e->shm16read_addr_next = 0xFFFFFFFF;
0701 e->shm32read_routing_next = 0xFFFFFFFF;
0702 e->shm32read_addr_next = 0xFFFFFFFF;
0703
0704 #define ADD_FILE(name, mode) \
0705 do { \
0706 debugfs_create_file(__stringify(name), \
0707 mode, e->subdir, dev, \
0708 &fops_##name.fops); \
0709 } while (0)
0710
0711
0712 ADD_FILE(shm16read, 0600);
0713 ADD_FILE(shm16write, 0200);
0714 ADD_FILE(shm32read, 0600);
0715 ADD_FILE(shm32write, 0200);
0716 ADD_FILE(mmio16read, 0600);
0717 ADD_FILE(mmio16write, 0200);
0718 ADD_FILE(mmio32read, 0600);
0719 ADD_FILE(mmio32write, 0200);
0720 ADD_FILE(txstat, 0400);
0721 ADD_FILE(restart, 0200);
0722 ADD_FILE(loctls, 0400);
0723
0724 #undef ADD_FILE
0725
0726 b43_add_dynamic_debug(dev);
0727 }
0728
0729 void b43_debugfs_remove_device(struct b43_wldev *dev)
0730 {
0731 struct b43_dfsentry *e;
0732
0733 if (!dev)
0734 return;
0735 e = dev->dfsentry;
0736 if (!e)
0737 return;
0738
0739 debugfs_remove(e->subdir);
0740 kfree(e->txstatlog.log);
0741 kfree(e);
0742 }
0743
0744 void b43_debugfs_log_txstat(struct b43_wldev *dev,
0745 const struct b43_txstatus *status)
0746 {
0747 struct b43_dfsentry *e = dev->dfsentry;
0748 struct b43_txstatus_log *log;
0749 struct b43_txstatus *cur;
0750 int i;
0751
0752 if (!e)
0753 return;
0754 log = &e->txstatlog;
0755 i = log->end + 1;
0756 if (i == B43_NR_LOGGED_TXSTATUS)
0757 i = 0;
0758 log->end = i;
0759 cur = &(log->log[i]);
0760 memcpy(cur, status, sizeof(*cur));
0761 }
0762
0763 void b43_debugfs_init(void)
0764 {
0765 rootdir = debugfs_create_dir(KBUILD_MODNAME, NULL);
0766 }
0767
0768 void b43_debugfs_exit(void)
0769 {
0770 debugfs_remove(rootdir);
0771 }