debug.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. /*
  2. * Copyright (c) 2012 Broadcom Corporation
  3. *
  4. * Permission to use, copy, modify, and/or distribute this software for any
  5. * purpose with or without fee is hereby granted, provided that the above
  6. * copyright notice and this permission notice appear in all copies.
  7. *
  8. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  9. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  10. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
  11. * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  12. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
  13. * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
  14. * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #include <linux/debugfs.h>
  17. #include <linux/netdevice.h>
  18. #include <linux/module.h>
  19. #include <brcmu_wifi.h>
  20. #include <brcmu_utils.h>
  21. #include "core.h"
  22. #include "bus.h"
  23. #include "debug.h"
  24. static struct dentry *root_folder;
  25. void brcmf_debugfs_init(void)
  26. {
  27. root_folder = debugfs_create_dir(KBUILD_MODNAME, NULL);
  28. if (IS_ERR(root_folder))
  29. root_folder = NULL;
  30. }
  31. void brcmf_debugfs_exit(void)
  32. {
  33. if (!root_folder)
  34. return;
  35. debugfs_remove_recursive(root_folder);
  36. root_folder = NULL;
  37. }
  38. static int brcmf_debugfs_chipinfo_read(struct seq_file *seq, void *data)
  39. {
  40. struct brcmf_bus *bus = dev_get_drvdata(seq->private);
  41. seq_printf(seq, "chip: %x(%u) rev %u\n",
  42. bus->chip, bus->chip, bus->chiprev);
  43. return 0;
  44. }
  45. int brcmf_debugfs_attach(struct brcmf_pub *drvr)
  46. {
  47. struct device *dev = drvr->bus_if->dev;
  48. if (!root_folder)
  49. return -ENODEV;
  50. drvr->dbgfs_dir = debugfs_create_dir(dev_name(dev), root_folder);
  51. brcmf_debugfs_add_entry(drvr, "chipinfo", brcmf_debugfs_chipinfo_read);
  52. return PTR_ERR_OR_ZERO(drvr->dbgfs_dir);
  53. }
  54. void brcmf_debugfs_detach(struct brcmf_pub *drvr)
  55. {
  56. if (!IS_ERR_OR_NULL(drvr->dbgfs_dir))
  57. debugfs_remove_recursive(drvr->dbgfs_dir);
  58. }
  59. struct dentry *brcmf_debugfs_get_devdir(struct brcmf_pub *drvr)
  60. {
  61. return drvr->dbgfs_dir;
  62. }
  63. struct brcmf_debugfs_entry {
  64. int (*read)(struct seq_file *seq, void *data);
  65. struct brcmf_pub *drvr;
  66. };
  67. static int brcmf_debugfs_entry_open(struct inode *inode, struct file *f)
  68. {
  69. struct brcmf_debugfs_entry *entry = inode->i_private;
  70. return single_open(f, entry->read, entry->drvr->bus_if->dev);
  71. }
  72. static const struct file_operations brcmf_debugfs_def_ops = {
  73. .owner = THIS_MODULE,
  74. .open = brcmf_debugfs_entry_open,
  75. .release = single_release,
  76. .read = seq_read,
  77. .llseek = seq_lseek
  78. };
  79. int brcmf_debugfs_add_entry(struct brcmf_pub *drvr, const char *fn,
  80. int (*read_fn)(struct seq_file *seq, void *data))
  81. {
  82. struct dentry *dentry = drvr->dbgfs_dir;
  83. struct brcmf_debugfs_entry *entry;
  84. if (IS_ERR_OR_NULL(dentry))
  85. return -ENOENT;
  86. entry = devm_kzalloc(drvr->bus_if->dev, sizeof(*entry), GFP_KERNEL);
  87. if (!entry)
  88. return -ENOMEM;
  89. entry->read = read_fn;
  90. entry->drvr = drvr;
  91. dentry = debugfs_create_file(fn, S_IRUGO, dentry, entry,
  92. &brcmf_debugfs_def_ops);
  93. return PTR_ERR_OR_ZERO(dentry);
  94. }