htc_drv_debug.c 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957
  1. /*
  2. * Copyright (c) 2010-2011 Atheros Communications Inc.
  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
  11. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  12. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  13. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  14. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #include "htc.h"
  17. static int ath9k_debugfs_open(struct inode *inode, struct file *file)
  18. {
  19. file->private_data = inode->i_private;
  20. return 0;
  21. }
  22. static ssize_t read_file_tgt_int_stats(struct file *file, char __user *user_buf,
  23. size_t count, loff_t *ppos)
  24. {
  25. struct ath9k_htc_priv *priv = file->private_data;
  26. struct ath9k_htc_target_int_stats cmd_rsp;
  27. char buf[512];
  28. unsigned int len = 0;
  29. int ret = 0;
  30. memset(&cmd_rsp, 0, sizeof(cmd_rsp));
  31. ath9k_htc_ps_wakeup(priv);
  32. WMI_CMD(WMI_INT_STATS_CMDID);
  33. if (ret) {
  34. ath9k_htc_ps_restore(priv);
  35. return -EINVAL;
  36. }
  37. ath9k_htc_ps_restore(priv);
  38. len += snprintf(buf + len, sizeof(buf) - len,
  39. "%20s : %10u\n", "RX",
  40. be32_to_cpu(cmd_rsp.rx));
  41. len += snprintf(buf + len, sizeof(buf) - len,
  42. "%20s : %10u\n", "RXORN",
  43. be32_to_cpu(cmd_rsp.rxorn));
  44. len += snprintf(buf + len, sizeof(buf) - len,
  45. "%20s : %10u\n", "RXEOL",
  46. be32_to_cpu(cmd_rsp.rxeol));
  47. len += snprintf(buf + len, sizeof(buf) - len,
  48. "%20s : %10u\n", "TXURN",
  49. be32_to_cpu(cmd_rsp.txurn));
  50. len += snprintf(buf + len, sizeof(buf) - len,
  51. "%20s : %10u\n", "TXTO",
  52. be32_to_cpu(cmd_rsp.txto));
  53. len += snprintf(buf + len, sizeof(buf) - len,
  54. "%20s : %10u\n", "CST",
  55. be32_to_cpu(cmd_rsp.cst));
  56. if (len > sizeof(buf))
  57. len = sizeof(buf);
  58. return simple_read_from_buffer(user_buf, count, ppos, buf, len);
  59. }
  60. static const struct file_operations fops_tgt_int_stats = {
  61. .read = read_file_tgt_int_stats,
  62. .open = ath9k_debugfs_open,
  63. .owner = THIS_MODULE,
  64. .llseek = default_llseek,
  65. };
  66. static ssize_t read_file_tgt_tx_stats(struct file *file, char __user *user_buf,
  67. size_t count, loff_t *ppos)
  68. {
  69. struct ath9k_htc_priv *priv = file->private_data;
  70. struct ath9k_htc_target_tx_stats cmd_rsp;
  71. char buf[512];
  72. unsigned int len = 0;
  73. int ret = 0;
  74. memset(&cmd_rsp, 0, sizeof(cmd_rsp));
  75. ath9k_htc_ps_wakeup(priv);
  76. WMI_CMD(WMI_TX_STATS_CMDID);
  77. if (ret) {
  78. ath9k_htc_ps_restore(priv);
  79. return -EINVAL;
  80. }
  81. ath9k_htc_ps_restore(priv);
  82. len += snprintf(buf + len, sizeof(buf) - len,
  83. "%20s : %10u\n", "Xretries",
  84. be32_to_cpu(cmd_rsp.xretries));
  85. len += snprintf(buf + len, sizeof(buf) - len,
  86. "%20s : %10u\n", "FifoErr",
  87. be32_to_cpu(cmd_rsp.fifoerr));
  88. len += snprintf(buf + len, sizeof(buf) - len,
  89. "%20s : %10u\n", "Filtered",
  90. be32_to_cpu(cmd_rsp.filtered));
  91. len += snprintf(buf + len, sizeof(buf) - len,
  92. "%20s : %10u\n", "TimerExp",
  93. be32_to_cpu(cmd_rsp.timer_exp));
  94. len += snprintf(buf + len, sizeof(buf) - len,
  95. "%20s : %10u\n", "ShortRetries",
  96. be32_to_cpu(cmd_rsp.shortretries));
  97. len += snprintf(buf + len, sizeof(buf) - len,
  98. "%20s : %10u\n", "LongRetries",
  99. be32_to_cpu(cmd_rsp.longretries));
  100. len += snprintf(buf + len, sizeof(buf) - len,
  101. "%20s : %10u\n", "QueueNull",
  102. be32_to_cpu(cmd_rsp.qnull));
  103. len += snprintf(buf + len, sizeof(buf) - len,
  104. "%20s : %10u\n", "EncapFail",
  105. be32_to_cpu(cmd_rsp.encap_fail));
  106. len += snprintf(buf + len, sizeof(buf) - len,
  107. "%20s : %10u\n", "NoBuf",
  108. be32_to_cpu(cmd_rsp.nobuf));
  109. if (len > sizeof(buf))
  110. len = sizeof(buf);
  111. return simple_read_from_buffer(user_buf, count, ppos, buf, len);
  112. }
  113. static const struct file_operations fops_tgt_tx_stats = {
  114. .read = read_file_tgt_tx_stats,
  115. .open = ath9k_debugfs_open,
  116. .owner = THIS_MODULE,
  117. .llseek = default_llseek,
  118. };
  119. static ssize_t read_file_tgt_rx_stats(struct file *file, char __user *user_buf,
  120. size_t count, loff_t *ppos)
  121. {
  122. struct ath9k_htc_priv *priv = file->private_data;
  123. struct ath9k_htc_target_rx_stats cmd_rsp;
  124. char buf[512];
  125. unsigned int len = 0;
  126. int ret = 0;
  127. memset(&cmd_rsp, 0, sizeof(cmd_rsp));
  128. ath9k_htc_ps_wakeup(priv);
  129. WMI_CMD(WMI_RX_STATS_CMDID);
  130. if (ret) {
  131. ath9k_htc_ps_restore(priv);
  132. return -EINVAL;
  133. }
  134. ath9k_htc_ps_restore(priv);
  135. len += snprintf(buf + len, sizeof(buf) - len,
  136. "%20s : %10u\n", "NoBuf",
  137. be32_to_cpu(cmd_rsp.nobuf));
  138. len += snprintf(buf + len, sizeof(buf) - len,
  139. "%20s : %10u\n", "HostSend",
  140. be32_to_cpu(cmd_rsp.host_send));
  141. len += snprintf(buf + len, sizeof(buf) - len,
  142. "%20s : %10u\n", "HostDone",
  143. be32_to_cpu(cmd_rsp.host_done));
  144. if (len > sizeof(buf))
  145. len = sizeof(buf);
  146. return simple_read_from_buffer(user_buf, count, ppos, buf, len);
  147. }
  148. static const struct file_operations fops_tgt_rx_stats = {
  149. .read = read_file_tgt_rx_stats,
  150. .open = ath9k_debugfs_open,
  151. .owner = THIS_MODULE,
  152. .llseek = default_llseek,
  153. };
  154. static ssize_t read_file_xmit(struct file *file, char __user *user_buf,
  155. size_t count, loff_t *ppos)
  156. {
  157. struct ath9k_htc_priv *priv = file->private_data;
  158. char buf[512];
  159. unsigned int len = 0;
  160. len += snprintf(buf + len, sizeof(buf) - len,
  161. "%20s : %10u\n", "Buffers queued",
  162. priv->debug.tx_stats.buf_queued);
  163. len += snprintf(buf + len, sizeof(buf) - len,
  164. "%20s : %10u\n", "Buffers completed",
  165. priv->debug.tx_stats.buf_completed);
  166. len += snprintf(buf + len, sizeof(buf) - len,
  167. "%20s : %10u\n", "SKBs queued",
  168. priv->debug.tx_stats.skb_queued);
  169. len += snprintf(buf + len, sizeof(buf) - len,
  170. "%20s : %10u\n", "SKBs success",
  171. priv->debug.tx_stats.skb_success);
  172. len += snprintf(buf + len, sizeof(buf) - len,
  173. "%20s : %10u\n", "SKBs failed",
  174. priv->debug.tx_stats.skb_failed);
  175. len += snprintf(buf + len, sizeof(buf) - len,
  176. "%20s : %10u\n", "CAB queued",
  177. priv->debug.tx_stats.cab_queued);
  178. len += snprintf(buf + len, sizeof(buf) - len,
  179. "%20s : %10u\n", "BE queued",
  180. priv->debug.tx_stats.queue_stats[WME_AC_BE]);
  181. len += snprintf(buf + len, sizeof(buf) - len,
  182. "%20s : %10u\n", "BK queued",
  183. priv->debug.tx_stats.queue_stats[WME_AC_BK]);
  184. len += snprintf(buf + len, sizeof(buf) - len,
  185. "%20s : %10u\n", "VI queued",
  186. priv->debug.tx_stats.queue_stats[WME_AC_VI]);
  187. len += snprintf(buf + len, sizeof(buf) - len,
  188. "%20s : %10u\n", "VO queued",
  189. priv->debug.tx_stats.queue_stats[WME_AC_VO]);
  190. if (len > sizeof(buf))
  191. len = sizeof(buf);
  192. return simple_read_from_buffer(user_buf, count, ppos, buf, len);
  193. }
  194. static const struct file_operations fops_xmit = {
  195. .read = read_file_xmit,
  196. .open = ath9k_debugfs_open,
  197. .owner = THIS_MODULE,
  198. .llseek = default_llseek,
  199. };
  200. void ath9k_htc_err_stat_rx(struct ath9k_htc_priv *priv,
  201. struct ath_htc_rx_status *rxs)
  202. {
  203. #define RX_PHY_ERR_INC(c) priv->debug.rx_stats.err_phy_stats[c]++
  204. if (rxs->rs_status & ATH9K_RXERR_CRC)
  205. priv->debug.rx_stats.err_crc++;
  206. if (rxs->rs_status & ATH9K_RXERR_DECRYPT)
  207. priv->debug.rx_stats.err_decrypt_crc++;
  208. if (rxs->rs_status & ATH9K_RXERR_MIC)
  209. priv->debug.rx_stats.err_mic++;
  210. if (rxs->rs_status & ATH9K_RX_DELIM_CRC_PRE)
  211. priv->debug.rx_stats.err_pre_delim++;
  212. if (rxs->rs_status & ATH9K_RX_DELIM_CRC_POST)
  213. priv->debug.rx_stats.err_post_delim++;
  214. if (rxs->rs_status & ATH9K_RX_DECRYPT_BUSY)
  215. priv->debug.rx_stats.err_decrypt_busy++;
  216. if (rxs->rs_status & ATH9K_RXERR_PHY) {
  217. priv->debug.rx_stats.err_phy++;
  218. if (rxs->rs_phyerr < ATH9K_PHYERR_MAX)
  219. RX_PHY_ERR_INC(rxs->rs_phyerr);
  220. }
  221. #undef RX_PHY_ERR_INC
  222. }
  223. static ssize_t read_file_recv(struct file *file, char __user *user_buf,
  224. size_t count, loff_t *ppos)
  225. {
  226. #define PHY_ERR(s, p) \
  227. len += snprintf(buf + len, size - len, "%20s : %10u\n", s, \
  228. priv->debug.rx_stats.err_phy_stats[p]);
  229. struct ath9k_htc_priv *priv = file->private_data;
  230. char *buf;
  231. unsigned int len = 0, size = 1500;
  232. ssize_t retval = 0;
  233. buf = kzalloc(size, GFP_KERNEL);
  234. if (buf == NULL)
  235. return -ENOMEM;
  236. len += snprintf(buf + len, size - len,
  237. "%20s : %10u\n", "SKBs allocated",
  238. priv->debug.rx_stats.skb_allocated);
  239. len += snprintf(buf + len, size - len,
  240. "%20s : %10u\n", "SKBs completed",
  241. priv->debug.rx_stats.skb_completed);
  242. len += snprintf(buf + len, size - len,
  243. "%20s : %10u\n", "SKBs Dropped",
  244. priv->debug.rx_stats.skb_dropped);
  245. len += snprintf(buf + len, size - len,
  246. "%20s : %10u\n", "CRC ERR",
  247. priv->debug.rx_stats.err_crc);
  248. len += snprintf(buf + len, size - len,
  249. "%20s : %10u\n", "DECRYPT CRC ERR",
  250. priv->debug.rx_stats.err_decrypt_crc);
  251. len += snprintf(buf + len, size - len,
  252. "%20s : %10u\n", "MIC ERR",
  253. priv->debug.rx_stats.err_mic);
  254. len += snprintf(buf + len, size - len,
  255. "%20s : %10u\n", "PRE-DELIM CRC ERR",
  256. priv->debug.rx_stats.err_pre_delim);
  257. len += snprintf(buf + len, size - len,
  258. "%20s : %10u\n", "POST-DELIM CRC ERR",
  259. priv->debug.rx_stats.err_post_delim);
  260. len += snprintf(buf + len, size - len,
  261. "%20s : %10u\n", "DECRYPT BUSY ERR",
  262. priv->debug.rx_stats.err_decrypt_busy);
  263. len += snprintf(buf + len, size - len,
  264. "%20s : %10u\n", "TOTAL PHY ERR",
  265. priv->debug.rx_stats.err_phy);
  266. PHY_ERR("UNDERRUN", ATH9K_PHYERR_UNDERRUN);
  267. PHY_ERR("TIMING", ATH9K_PHYERR_TIMING);
  268. PHY_ERR("PARITY", ATH9K_PHYERR_PARITY);
  269. PHY_ERR("RATE", ATH9K_PHYERR_RATE);
  270. PHY_ERR("LENGTH", ATH9K_PHYERR_LENGTH);
  271. PHY_ERR("RADAR", ATH9K_PHYERR_RADAR);
  272. PHY_ERR("SERVICE", ATH9K_PHYERR_SERVICE);
  273. PHY_ERR("TOR", ATH9K_PHYERR_TOR);
  274. PHY_ERR("OFDM-TIMING", ATH9K_PHYERR_OFDM_TIMING);
  275. PHY_ERR("OFDM-SIGNAL-PARITY", ATH9K_PHYERR_OFDM_SIGNAL_PARITY);
  276. PHY_ERR("OFDM-RATE", ATH9K_PHYERR_OFDM_RATE_ILLEGAL);
  277. PHY_ERR("OFDM-LENGTH", ATH9K_PHYERR_OFDM_LENGTH_ILLEGAL);
  278. PHY_ERR("OFDM-POWER-DROP", ATH9K_PHYERR_OFDM_POWER_DROP);
  279. PHY_ERR("OFDM-SERVICE", ATH9K_PHYERR_OFDM_SERVICE);
  280. PHY_ERR("OFDM-RESTART", ATH9K_PHYERR_OFDM_RESTART);
  281. PHY_ERR("FALSE-RADAR-EXT", ATH9K_PHYERR_FALSE_RADAR_EXT);
  282. PHY_ERR("CCK-TIMING", ATH9K_PHYERR_CCK_TIMING);
  283. PHY_ERR("CCK-HEADER-CRC", ATH9K_PHYERR_CCK_HEADER_CRC);
  284. PHY_ERR("CCK-RATE", ATH9K_PHYERR_CCK_RATE_ILLEGAL);
  285. PHY_ERR("CCK-SERVICE", ATH9K_PHYERR_CCK_SERVICE);
  286. PHY_ERR("CCK-RESTART", ATH9K_PHYERR_CCK_RESTART);
  287. PHY_ERR("CCK-LENGTH", ATH9K_PHYERR_CCK_LENGTH_ILLEGAL);
  288. PHY_ERR("CCK-POWER-DROP", ATH9K_PHYERR_CCK_POWER_DROP);
  289. PHY_ERR("HT-CRC", ATH9K_PHYERR_HT_CRC_ERROR);
  290. PHY_ERR("HT-LENGTH", ATH9K_PHYERR_HT_LENGTH_ILLEGAL);
  291. PHY_ERR("HT-RATE", ATH9K_PHYERR_HT_RATE_ILLEGAL);
  292. if (len > size)
  293. len = size;
  294. retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
  295. kfree(buf);
  296. return retval;
  297. #undef PHY_ERR
  298. }
  299. static const struct file_operations fops_recv = {
  300. .read = read_file_recv,
  301. .open = ath9k_debugfs_open,
  302. .owner = THIS_MODULE,
  303. .llseek = default_llseek,
  304. };
  305. static ssize_t read_file_slot(struct file *file, char __user *user_buf,
  306. size_t count, loff_t *ppos)
  307. {
  308. struct ath9k_htc_priv *priv = file->private_data;
  309. char buf[512];
  310. unsigned int len = 0;
  311. spin_lock_bh(&priv->tx.tx_lock);
  312. len += snprintf(buf + len, sizeof(buf) - len, "TX slot bitmap : ");
  313. len += bitmap_scnprintf(buf + len, sizeof(buf) - len,
  314. priv->tx.tx_slot, MAX_TX_BUF_NUM);
  315. len += snprintf(buf + len, sizeof(buf) - len, "\n");
  316. len += snprintf(buf + len, sizeof(buf) - len,
  317. "Used slots : %d\n",
  318. bitmap_weight(priv->tx.tx_slot, MAX_TX_BUF_NUM));
  319. spin_unlock_bh(&priv->tx.tx_lock);
  320. if (len > sizeof(buf))
  321. len = sizeof(buf);
  322. return simple_read_from_buffer(user_buf, count, ppos, buf, len);
  323. }
  324. static const struct file_operations fops_slot = {
  325. .read = read_file_slot,
  326. .open = ath9k_debugfs_open,
  327. .owner = THIS_MODULE,
  328. .llseek = default_llseek,
  329. };
  330. static ssize_t read_file_queue(struct file *file, char __user *user_buf,
  331. size_t count, loff_t *ppos)
  332. {
  333. struct ath9k_htc_priv *priv = file->private_data;
  334. char buf[512];
  335. unsigned int len = 0;
  336. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  337. "Mgmt endpoint", skb_queue_len(&priv->tx.mgmt_ep_queue));
  338. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  339. "Cab endpoint", skb_queue_len(&priv->tx.cab_ep_queue));
  340. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  341. "Data BE endpoint", skb_queue_len(&priv->tx.data_be_queue));
  342. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  343. "Data BK endpoint", skb_queue_len(&priv->tx.data_bk_queue));
  344. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  345. "Data VI endpoint", skb_queue_len(&priv->tx.data_vi_queue));
  346. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  347. "Data VO endpoint", skb_queue_len(&priv->tx.data_vo_queue));
  348. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  349. "Failed queue", skb_queue_len(&priv->tx.tx_failed));
  350. spin_lock_bh(&priv->tx.tx_lock);
  351. len += snprintf(buf + len, sizeof(buf) - len, "%20s : %10u\n",
  352. "Queued count", priv->tx.queued_cnt);
  353. spin_unlock_bh(&priv->tx.tx_lock);
  354. if (len > sizeof(buf))
  355. len = sizeof(buf);
  356. return simple_read_from_buffer(user_buf, count, ppos, buf, len);
  357. }
  358. static const struct file_operations fops_queue = {
  359. .read = read_file_queue,
  360. .open = ath9k_debugfs_open,
  361. .owner = THIS_MODULE,
  362. .llseek = default_llseek,
  363. };
  364. static ssize_t read_file_debug(struct file *file, char __user *user_buf,
  365. size_t count, loff_t *ppos)
  366. {
  367. struct ath9k_htc_priv *priv = file->private_data;
  368. struct ath_common *common = ath9k_hw_common(priv->ah);
  369. char buf[32];
  370. unsigned int len;
  371. len = sprintf(buf, "0x%08x\n", common->debug_mask);
  372. return simple_read_from_buffer(user_buf, count, ppos, buf, len);
  373. }
  374. static ssize_t write_file_debug(struct file *file, const char __user *user_buf,
  375. size_t count, loff_t *ppos)
  376. {
  377. struct ath9k_htc_priv *priv = file->private_data;
  378. struct ath_common *common = ath9k_hw_common(priv->ah);
  379. unsigned long mask;
  380. char buf[32];
  381. ssize_t len;
  382. len = min(count, sizeof(buf) - 1);
  383. if (copy_from_user(buf, user_buf, len))
  384. return -EFAULT;
  385. buf[len] = '\0';
  386. if (strict_strtoul(buf, 0, &mask))
  387. return -EINVAL;
  388. common->debug_mask = mask;
  389. return count;
  390. }
  391. static const struct file_operations fops_debug = {
  392. .read = read_file_debug,
  393. .write = write_file_debug,
  394. .open = ath9k_debugfs_open,
  395. .owner = THIS_MODULE,
  396. .llseek = default_llseek,
  397. };
  398. static ssize_t read_file_base_eeprom(struct file *file, char __user *user_buf,
  399. size_t count, loff_t *ppos)
  400. {
  401. struct ath9k_htc_priv *priv = file->private_data;
  402. struct ath_common *common = ath9k_hw_common(priv->ah);
  403. struct base_eep_header *pBase = NULL;
  404. unsigned int len = 0, size = 1500;
  405. ssize_t retval = 0;
  406. char *buf;
  407. /*
  408. * This can be done since all the 3 EEPROM families have the
  409. * same base header upto a certain point, and we are interested in
  410. * the data only upto that point.
  411. */
  412. if (AR_SREV_9271(priv->ah))
  413. pBase = (struct base_eep_header *)
  414. &priv->ah->eeprom.map4k.baseEepHeader;
  415. else if (priv->ah->hw_version.usbdev == AR9280_USB)
  416. pBase = (struct base_eep_header *)
  417. &priv->ah->eeprom.def.baseEepHeader;
  418. else if (priv->ah->hw_version.usbdev == AR9287_USB)
  419. pBase = (struct base_eep_header *)
  420. &priv->ah->eeprom.map9287.baseEepHeader;
  421. if (pBase == NULL) {
  422. ath_err(common, "Unknown EEPROM type\n");
  423. return 0;
  424. }
  425. buf = kzalloc(size, GFP_KERNEL);
  426. if (buf == NULL)
  427. return -ENOMEM;
  428. len += snprintf(buf + len, size - len,
  429. "%20s : %10d\n", "Major Version",
  430. pBase->version >> 12);
  431. len += snprintf(buf + len, size - len,
  432. "%20s : %10d\n", "Minor Version",
  433. pBase->version & 0xFFF);
  434. len += snprintf(buf + len, size - len,
  435. "%20s : %10d\n", "Checksum",
  436. pBase->checksum);
  437. len += snprintf(buf + len, size - len,
  438. "%20s : %10d\n", "Length",
  439. pBase->length);
  440. len += snprintf(buf + len, size - len,
  441. "%20s : %10d\n", "RegDomain1",
  442. pBase->regDmn[0]);
  443. len += snprintf(buf + len, size - len,
  444. "%20s : %10d\n", "RegDomain2",
  445. pBase->regDmn[1]);
  446. len += snprintf(buf + len, size - len,
  447. "%20s : %10d\n",
  448. "TX Mask", pBase->txMask);
  449. len += snprintf(buf + len, size - len,
  450. "%20s : %10d\n",
  451. "RX Mask", pBase->rxMask);
  452. len += snprintf(buf + len, size - len,
  453. "%20s : %10d\n",
  454. "Allow 5GHz",
  455. !!(pBase->opCapFlags & AR5416_OPFLAGS_11A));
  456. len += snprintf(buf + len, size - len,
  457. "%20s : %10d\n",
  458. "Allow 2GHz",
  459. !!(pBase->opCapFlags & AR5416_OPFLAGS_11G));
  460. len += snprintf(buf + len, size - len,
  461. "%20s : %10d\n",
  462. "Disable 2GHz HT20",
  463. !!(pBase->opCapFlags & AR5416_OPFLAGS_N_2G_HT20));
  464. len += snprintf(buf + len, size - len,
  465. "%20s : %10d\n",
  466. "Disable 2GHz HT40",
  467. !!(pBase->opCapFlags & AR5416_OPFLAGS_N_2G_HT40));
  468. len += snprintf(buf + len, size - len,
  469. "%20s : %10d\n",
  470. "Disable 5Ghz HT20",
  471. !!(pBase->opCapFlags & AR5416_OPFLAGS_N_5G_HT20));
  472. len += snprintf(buf + len, size - len,
  473. "%20s : %10d\n",
  474. "Disable 5Ghz HT40",
  475. !!(pBase->opCapFlags & AR5416_OPFLAGS_N_5G_HT40));
  476. len += snprintf(buf + len, size - len,
  477. "%20s : %10d\n",
  478. "Big Endian",
  479. !!(pBase->eepMisc & 0x01));
  480. len += snprintf(buf + len, size - len,
  481. "%20s : %10d\n",
  482. "Cal Bin Major Ver",
  483. (pBase->binBuildNumber >> 24) & 0xFF);
  484. len += snprintf(buf + len, size - len,
  485. "%20s : %10d\n",
  486. "Cal Bin Minor Ver",
  487. (pBase->binBuildNumber >> 16) & 0xFF);
  488. len += snprintf(buf + len, size - len,
  489. "%20s : %10d\n",
  490. "Cal Bin Build",
  491. (pBase->binBuildNumber >> 8) & 0xFF);
  492. /*
  493. * UB91 specific data.
  494. */
  495. if (AR_SREV_9271(priv->ah)) {
  496. struct base_eep_header_4k *pBase4k =
  497. &priv->ah->eeprom.map4k.baseEepHeader;
  498. len += snprintf(buf + len, size - len,
  499. "%20s : %10d\n",
  500. "TX Gain type",
  501. pBase4k->txGainType);
  502. }
  503. /*
  504. * UB95 specific data.
  505. */
  506. if (priv->ah->hw_version.usbdev == AR9287_USB) {
  507. struct base_eep_ar9287_header *pBase9287 =
  508. &priv->ah->eeprom.map9287.baseEepHeader;
  509. len += snprintf(buf + len, size - len,
  510. "%20s : %10ddB\n",
  511. "Power Table Offset",
  512. pBase9287->pwrTableOffset);
  513. len += snprintf(buf + len, size - len,
  514. "%20s : %10d\n",
  515. "OpenLoop Power Ctrl",
  516. pBase9287->openLoopPwrCntl);
  517. }
  518. len += snprintf(buf + len, size - len, "%20s : %pM\n", "MacAddress",
  519. pBase->macAddr);
  520. if (len > size)
  521. len = size;
  522. retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
  523. kfree(buf);
  524. return retval;
  525. }
  526. static const struct file_operations fops_base_eeprom = {
  527. .read = read_file_base_eeprom,
  528. .open = ath9k_debugfs_open,
  529. .owner = THIS_MODULE,
  530. .llseek = default_llseek,
  531. };
  532. static ssize_t read_4k_modal_eeprom(struct file *file,
  533. char __user *user_buf,
  534. size_t count, loff_t *ppos)
  535. {
  536. #define PR_EEP(_s, _val) \
  537. do { \
  538. len += snprintf(buf + len, size - len, "%20s : %10d\n", \
  539. _s, (_val)); \
  540. } while (0)
  541. struct ath9k_htc_priv *priv = file->private_data;
  542. struct modal_eep_4k_header *pModal = &priv->ah->eeprom.map4k.modalHeader;
  543. unsigned int len = 0, size = 2048;
  544. ssize_t retval = 0;
  545. char *buf;
  546. buf = kzalloc(size, GFP_KERNEL);
  547. if (buf == NULL)
  548. return -ENOMEM;
  549. PR_EEP("Chain0 Ant. Control", pModal->antCtrlChain[0]);
  550. PR_EEP("Ant. Common Control", pModal->antCtrlCommon);
  551. PR_EEP("Chain0 Ant. Gain", pModal->antennaGainCh[0]);
  552. PR_EEP("Switch Settle", pModal->switchSettling);
  553. PR_EEP("Chain0 TxRxAtten", pModal->txRxAttenCh[0]);
  554. PR_EEP("Chain0 RxTxMargin", pModal->rxTxMarginCh[0]);
  555. PR_EEP("ADC Desired size", pModal->adcDesiredSize);
  556. PR_EEP("PGA Desired size", pModal->pgaDesiredSize);
  557. PR_EEP("Chain0 xlna Gain", pModal->xlnaGainCh[0]);
  558. PR_EEP("txEndToXpaOff", pModal->txEndToXpaOff);
  559. PR_EEP("txEndToRxOn", pModal->txEndToRxOn);
  560. PR_EEP("txFrameToXpaOn", pModal->txFrameToXpaOn);
  561. PR_EEP("CCA Threshold)", pModal->thresh62);
  562. PR_EEP("Chain0 NF Threshold", pModal->noiseFloorThreshCh[0]);
  563. PR_EEP("xpdGain", pModal->xpdGain);
  564. PR_EEP("External PD", pModal->xpd);
  565. PR_EEP("Chain0 I Coefficient", pModal->iqCalICh[0]);
  566. PR_EEP("Chain0 Q Coefficient", pModal->iqCalQCh[0]);
  567. PR_EEP("pdGainOverlap", pModal->pdGainOverlap);
  568. PR_EEP("O/D Bias Version", pModal->version);
  569. PR_EEP("CCK OutputBias", pModal->ob_0);
  570. PR_EEP("BPSK OutputBias", pModal->ob_1);
  571. PR_EEP("QPSK OutputBias", pModal->ob_2);
  572. PR_EEP("16QAM OutputBias", pModal->ob_3);
  573. PR_EEP("64QAM OutputBias", pModal->ob_4);
  574. PR_EEP("CCK Driver1_Bias", pModal->db1_0);
  575. PR_EEP("BPSK Driver1_Bias", pModal->db1_1);
  576. PR_EEP("QPSK Driver1_Bias", pModal->db1_2);
  577. PR_EEP("16QAM Driver1_Bias", pModal->db1_3);
  578. PR_EEP("64QAM Driver1_Bias", pModal->db1_4);
  579. PR_EEP("CCK Driver2_Bias", pModal->db2_0);
  580. PR_EEP("BPSK Driver2_Bias", pModal->db2_1);
  581. PR_EEP("QPSK Driver2_Bias", pModal->db2_2);
  582. PR_EEP("16QAM Driver2_Bias", pModal->db2_3);
  583. PR_EEP("64QAM Driver2_Bias", pModal->db2_4);
  584. PR_EEP("xPA Bias Level", pModal->xpaBiasLvl);
  585. PR_EEP("txFrameToDataStart", pModal->txFrameToDataStart);
  586. PR_EEP("txFrameToPaOn", pModal->txFrameToPaOn);
  587. PR_EEP("HT40 Power Inc.", pModal->ht40PowerIncForPdadc);
  588. PR_EEP("Chain0 bswAtten", pModal->bswAtten[0]);
  589. PR_EEP("Chain0 bswMargin", pModal->bswMargin[0]);
  590. PR_EEP("HT40 Switch Settle", pModal->swSettleHt40);
  591. PR_EEP("Chain0 xatten2Db", pModal->xatten2Db[0]);
  592. PR_EEP("Chain0 xatten2Margin", pModal->xatten2Margin[0]);
  593. PR_EEP("Ant. Diversity ctl1", pModal->antdiv_ctl1);
  594. PR_EEP("Ant. Diversity ctl2", pModal->antdiv_ctl2);
  595. PR_EEP("TX Diversity", pModal->tx_diversity);
  596. if (len > size)
  597. len = size;
  598. retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
  599. kfree(buf);
  600. return retval;
  601. #undef PR_EEP
  602. }
  603. static ssize_t read_def_modal_eeprom(struct file *file,
  604. char __user *user_buf,
  605. size_t count, loff_t *ppos)
  606. {
  607. #define PR_EEP(_s, _val) \
  608. do { \
  609. if (pBase->opCapFlags & AR5416_OPFLAGS_11G) { \
  610. pModal = &priv->ah->eeprom.def.modalHeader[1]; \
  611. len += snprintf(buf + len, size - len, "%20s : %8d%7s", \
  612. _s, (_val), "|"); \
  613. } \
  614. if (pBase->opCapFlags & AR5416_OPFLAGS_11A) { \
  615. pModal = &priv->ah->eeprom.def.modalHeader[0]; \
  616. len += snprintf(buf + len, size - len, "%9d\n", \
  617. (_val)); \
  618. } \
  619. } while (0)
  620. struct ath9k_htc_priv *priv = file->private_data;
  621. struct base_eep_header *pBase = &priv->ah->eeprom.def.baseEepHeader;
  622. struct modal_eep_header *pModal = NULL;
  623. unsigned int len = 0, size = 3500;
  624. ssize_t retval = 0;
  625. char *buf;
  626. buf = kzalloc(size, GFP_KERNEL);
  627. if (buf == NULL)
  628. return -ENOMEM;
  629. len += snprintf(buf + len, size - len,
  630. "%31s %15s\n", "2G", "5G");
  631. len += snprintf(buf + len, size - len,
  632. "%32s %16s\n", "====", "====\n");
  633. PR_EEP("Chain0 Ant. Control", pModal->antCtrlChain[0]);
  634. PR_EEP("Chain1 Ant. Control", pModal->antCtrlChain[1]);
  635. PR_EEP("Chain2 Ant. Control", pModal->antCtrlChain[2]);
  636. PR_EEP("Ant. Common Control", pModal->antCtrlCommon);
  637. PR_EEP("Chain0 Ant. Gain", pModal->antennaGainCh[0]);
  638. PR_EEP("Chain1 Ant. Gain", pModal->antennaGainCh[1]);
  639. PR_EEP("Chain2 Ant. Gain", pModal->antennaGainCh[2]);
  640. PR_EEP("Switch Settle", pModal->switchSettling);
  641. PR_EEP("Chain0 TxRxAtten", pModal->txRxAttenCh[0]);
  642. PR_EEP("Chain1 TxRxAtten", pModal->txRxAttenCh[1]);
  643. PR_EEP("Chain2 TxRxAtten", pModal->txRxAttenCh[2]);
  644. PR_EEP("Chain0 RxTxMargin", pModal->rxTxMarginCh[0]);
  645. PR_EEP("Chain1 RxTxMargin", pModal->rxTxMarginCh[1]);
  646. PR_EEP("Chain2 RxTxMargin", pModal->rxTxMarginCh[2]);
  647. PR_EEP("ADC Desired size", pModal->adcDesiredSize);
  648. PR_EEP("PGA Desired size", pModal->pgaDesiredSize);
  649. PR_EEP("Chain0 xlna Gain", pModal->xlnaGainCh[0]);
  650. PR_EEP("Chain1 xlna Gain", pModal->xlnaGainCh[1]);
  651. PR_EEP("Chain2 xlna Gain", pModal->xlnaGainCh[2]);
  652. PR_EEP("txEndToXpaOff", pModal->txEndToXpaOff);
  653. PR_EEP("txEndToRxOn", pModal->txEndToRxOn);
  654. PR_EEP("txFrameToXpaOn", pModal->txFrameToXpaOn);
  655. PR_EEP("CCA Threshold)", pModal->thresh62);
  656. PR_EEP("Chain0 NF Threshold", pModal->noiseFloorThreshCh[0]);
  657. PR_EEP("Chain1 NF Threshold", pModal->noiseFloorThreshCh[1]);
  658. PR_EEP("Chain2 NF Threshold", pModal->noiseFloorThreshCh[2]);
  659. PR_EEP("xpdGain", pModal->xpdGain);
  660. PR_EEP("External PD", pModal->xpd);
  661. PR_EEP("Chain0 I Coefficient", pModal->iqCalICh[0]);
  662. PR_EEP("Chain1 I Coefficient", pModal->iqCalICh[1]);
  663. PR_EEP("Chain2 I Coefficient", pModal->iqCalICh[2]);
  664. PR_EEP("Chain0 Q Coefficient", pModal->iqCalQCh[0]);
  665. PR_EEP("Chain1 Q Coefficient", pModal->iqCalQCh[1]);
  666. PR_EEP("Chain2 Q Coefficient", pModal->iqCalQCh[2]);
  667. PR_EEP("pdGainOverlap", pModal->pdGainOverlap);
  668. PR_EEP("Chain0 OutputBias", pModal->ob);
  669. PR_EEP("Chain0 DriverBias", pModal->db);
  670. PR_EEP("xPA Bias Level", pModal->xpaBiasLvl);
  671. PR_EEP("2chain pwr decrease", pModal->pwrDecreaseFor2Chain);
  672. PR_EEP("3chain pwr decrease", pModal->pwrDecreaseFor3Chain);
  673. PR_EEP("txFrameToDataStart", pModal->txFrameToDataStart);
  674. PR_EEP("txFrameToPaOn", pModal->txFrameToPaOn);
  675. PR_EEP("HT40 Power Inc.", pModal->ht40PowerIncForPdadc);
  676. PR_EEP("Chain0 bswAtten", pModal->bswAtten[0]);
  677. PR_EEP("Chain1 bswAtten", pModal->bswAtten[1]);
  678. PR_EEP("Chain2 bswAtten", pModal->bswAtten[2]);
  679. PR_EEP("Chain0 bswMargin", pModal->bswMargin[0]);
  680. PR_EEP("Chain1 bswMargin", pModal->bswMargin[1]);
  681. PR_EEP("Chain2 bswMargin", pModal->bswMargin[2]);
  682. PR_EEP("HT40 Switch Settle", pModal->swSettleHt40);
  683. PR_EEP("Chain0 xatten2Db", pModal->xatten2Db[0]);
  684. PR_EEP("Chain1 xatten2Db", pModal->xatten2Db[1]);
  685. PR_EEP("Chain2 xatten2Db", pModal->xatten2Db[2]);
  686. PR_EEP("Chain0 xatten2Margin", pModal->xatten2Margin[0]);
  687. PR_EEP("Chain1 xatten2Margin", pModal->xatten2Margin[1]);
  688. PR_EEP("Chain2 xatten2Margin", pModal->xatten2Margin[2]);
  689. PR_EEP("Chain1 OutputBias", pModal->ob_ch1);
  690. PR_EEP("Chain1 DriverBias", pModal->db_ch1);
  691. PR_EEP("LNA Control", pModal->lna_ctl);
  692. PR_EEP("XPA Bias Freq0", pModal->xpaBiasLvlFreq[0]);
  693. PR_EEP("XPA Bias Freq1", pModal->xpaBiasLvlFreq[1]);
  694. PR_EEP("XPA Bias Freq2", pModal->xpaBiasLvlFreq[2]);
  695. if (len > size)
  696. len = size;
  697. retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
  698. kfree(buf);
  699. return retval;
  700. #undef PR_EEP
  701. }
  702. static ssize_t read_9287_modal_eeprom(struct file *file,
  703. char __user *user_buf,
  704. size_t count, loff_t *ppos)
  705. {
  706. #define PR_EEP(_s, _val) \
  707. do { \
  708. len += snprintf(buf + len, size - len, "%20s : %10d\n", \
  709. _s, (_val)); \
  710. } while (0)
  711. struct ath9k_htc_priv *priv = file->private_data;
  712. struct modal_eep_ar9287_header *pModal = &priv->ah->eeprom.map9287.modalHeader;
  713. unsigned int len = 0, size = 3000;
  714. ssize_t retval = 0;
  715. char *buf;
  716. buf = kzalloc(size, GFP_KERNEL);
  717. if (buf == NULL)
  718. return -ENOMEM;
  719. PR_EEP("Chain0 Ant. Control", pModal->antCtrlChain[0]);
  720. PR_EEP("Chain1 Ant. Control", pModal->antCtrlChain[1]);
  721. PR_EEP("Ant. Common Control", pModal->antCtrlCommon);
  722. PR_EEP("Chain0 Ant. Gain", pModal->antennaGainCh[0]);
  723. PR_EEP("Chain1 Ant. Gain", pModal->antennaGainCh[1]);
  724. PR_EEP("Switch Settle", pModal->switchSettling);
  725. PR_EEP("Chain0 TxRxAtten", pModal->txRxAttenCh[0]);
  726. PR_EEP("Chain1 TxRxAtten", pModal->txRxAttenCh[1]);
  727. PR_EEP("Chain0 RxTxMargin", pModal->rxTxMarginCh[0]);
  728. PR_EEP("Chain1 RxTxMargin", pModal->rxTxMarginCh[1]);
  729. PR_EEP("ADC Desired size", pModal->adcDesiredSize);
  730. PR_EEP("txEndToXpaOff", pModal->txEndToXpaOff);
  731. PR_EEP("txEndToRxOn", pModal->txEndToRxOn);
  732. PR_EEP("txFrameToXpaOn", pModal->txFrameToXpaOn);
  733. PR_EEP("CCA Threshold)", pModal->thresh62);
  734. PR_EEP("Chain0 NF Threshold", pModal->noiseFloorThreshCh[0]);
  735. PR_EEP("Chain1 NF Threshold", pModal->noiseFloorThreshCh[1]);
  736. PR_EEP("xpdGain", pModal->xpdGain);
  737. PR_EEP("External PD", pModal->xpd);
  738. PR_EEP("Chain0 I Coefficient", pModal->iqCalICh[0]);
  739. PR_EEP("Chain1 I Coefficient", pModal->iqCalICh[1]);
  740. PR_EEP("Chain0 Q Coefficient", pModal->iqCalQCh[0]);
  741. PR_EEP("Chain1 Q Coefficient", pModal->iqCalQCh[1]);
  742. PR_EEP("pdGainOverlap", pModal->pdGainOverlap);
  743. PR_EEP("xPA Bias Level", pModal->xpaBiasLvl);
  744. PR_EEP("txFrameToDataStart", pModal->txFrameToDataStart);
  745. PR_EEP("txFrameToPaOn", pModal->txFrameToPaOn);
  746. PR_EEP("HT40 Power Inc.", pModal->ht40PowerIncForPdadc);
  747. PR_EEP("Chain0 bswAtten", pModal->bswAtten[0]);
  748. PR_EEP("Chain1 bswAtten", pModal->bswAtten[1]);
  749. PR_EEP("Chain0 bswMargin", pModal->bswMargin[0]);
  750. PR_EEP("Chain1 bswMargin", pModal->bswMargin[1]);
  751. PR_EEP("HT40 Switch Settle", pModal->swSettleHt40);
  752. PR_EEP("AR92x7 Version", pModal->version);
  753. PR_EEP("DriverBias1", pModal->db1);
  754. PR_EEP("DriverBias2", pModal->db1);
  755. PR_EEP("CCK OutputBias", pModal->ob_cck);
  756. PR_EEP("PSK OutputBias", pModal->ob_psk);
  757. PR_EEP("QAM OutputBias", pModal->ob_qam);
  758. PR_EEP("PAL_OFF OutputBias", pModal->ob_pal_off);
  759. if (len > size)
  760. len = size;
  761. retval = simple_read_from_buffer(user_buf, count, ppos, buf, len);
  762. kfree(buf);
  763. return retval;
  764. #undef PR_EEP
  765. }
  766. static ssize_t read_file_modal_eeprom(struct file *file, char __user *user_buf,
  767. size_t count, loff_t *ppos)
  768. {
  769. struct ath9k_htc_priv *priv = file->private_data;
  770. if (AR_SREV_9271(priv->ah))
  771. return read_4k_modal_eeprom(file, user_buf, count, ppos);
  772. else if (priv->ah->hw_version.usbdev == AR9280_USB)
  773. return read_def_modal_eeprom(file, user_buf, count, ppos);
  774. else if (priv->ah->hw_version.usbdev == AR9287_USB)
  775. return read_9287_modal_eeprom(file, user_buf, count, ppos);
  776. return 0;
  777. }
  778. static const struct file_operations fops_modal_eeprom = {
  779. .read = read_file_modal_eeprom,
  780. .open = ath9k_debugfs_open,
  781. .owner = THIS_MODULE,
  782. .llseek = default_llseek,
  783. };
  784. int ath9k_htc_init_debug(struct ath_hw *ah)
  785. {
  786. struct ath_common *common = ath9k_hw_common(ah);
  787. struct ath9k_htc_priv *priv = (struct ath9k_htc_priv *) common->priv;
  788. priv->debug.debugfs_phy = debugfs_create_dir(KBUILD_MODNAME,
  789. priv->hw->wiphy->debugfsdir);
  790. if (!priv->debug.debugfs_phy)
  791. return -ENOMEM;
  792. debugfs_create_file("tgt_int_stats", S_IRUSR, priv->debug.debugfs_phy,
  793. priv, &fops_tgt_int_stats);
  794. debugfs_create_file("tgt_tx_stats", S_IRUSR, priv->debug.debugfs_phy,
  795. priv, &fops_tgt_tx_stats);
  796. debugfs_create_file("tgt_rx_stats", S_IRUSR, priv->debug.debugfs_phy,
  797. priv, &fops_tgt_rx_stats);
  798. debugfs_create_file("xmit", S_IRUSR, priv->debug.debugfs_phy,
  799. priv, &fops_xmit);
  800. debugfs_create_file("recv", S_IRUSR, priv->debug.debugfs_phy,
  801. priv, &fops_recv);
  802. debugfs_create_file("slot", S_IRUSR, priv->debug.debugfs_phy,
  803. priv, &fops_slot);
  804. debugfs_create_file("queue", S_IRUSR, priv->debug.debugfs_phy,
  805. priv, &fops_queue);
  806. debugfs_create_file("debug", S_IRUSR | S_IWUSR, priv->debug.debugfs_phy,
  807. priv, &fops_debug);
  808. debugfs_create_file("base_eeprom", S_IRUSR, priv->debug.debugfs_phy,
  809. priv, &fops_base_eeprom);
  810. debugfs_create_file("modal_eeprom", S_IRUSR, priv->debug.debugfs_phy,
  811. priv, &fops_modal_eeprom);
  812. return 0;
  813. }