debug.c 31 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /* Copyright (c) 2016-2021, The Linux Foundation. All rights reserved. */
  3. /* Copyright (c) 2022-2023 Qualcomm Innovation Center, Inc. All rights reserved. */
  4. #include <linux/err.h>
  5. #include <linux/seq_file.h>
  6. #include <linux/debugfs.h>
  7. #include "main.h"
  8. #include "bus.h"
  9. #include "debug.h"
  10. #include "pci.h"
  11. #define MMIO_REG_ACCESS_MEM_TYPE 0xFF
  12. #define MMIO_REG_RAW_ACCESS_MEM_TYPE 0xFE
  13. #define DEFAULT_KERNEL_LOG_LEVEL INFO_LOG
  14. #define DEFAULT_IPC_LOG_LEVEL DEBUG_LOG
  15. enum log_level cnss_kernel_log_level = DEFAULT_KERNEL_LOG_LEVEL;
  16. #if IS_ENABLED(CONFIG_IPC_LOGGING)
  17. void *cnss_ipc_log_context;
  18. void *cnss_ipc_log_long_context;
  19. enum log_level cnss_ipc_log_level = DEFAULT_IPC_LOG_LEVEL;
  20. static int cnss_set_ipc_log_level(u32 val)
  21. {
  22. if (val < MAX_LOG) {
  23. cnss_ipc_log_level = val;
  24. return 0;
  25. }
  26. return -EINVAL;
  27. }
  28. static u32 cnss_get_ipc_log_level(void)
  29. {
  30. return cnss_ipc_log_level;
  31. }
  32. #else
  33. static int cnss_set_ipc_log_level(int val) { return -EINVAL; }
  34. static u32 cnss_get_ipc_log_level(void) { return MAX_LOG; }
  35. #endif
  36. static int cnss_pin_connect_show(struct seq_file *s, void *data)
  37. {
  38. struct cnss_plat_data *cnss_priv = s->private;
  39. seq_puts(s, "Pin connect results\n");
  40. seq_printf(s, "FW power pin result: %04x\n",
  41. cnss_priv->pin_result.fw_pwr_pin_result);
  42. seq_printf(s, "FW PHY IO pin result: %04x\n",
  43. cnss_priv->pin_result.fw_phy_io_pin_result);
  44. seq_printf(s, "FW RF pin result: %04x\n",
  45. cnss_priv->pin_result.fw_rf_pin_result);
  46. seq_printf(s, "Host pin result: %04x\n",
  47. cnss_priv->pin_result.host_pin_result);
  48. seq_puts(s, "\n");
  49. return 0;
  50. }
  51. static int cnss_pin_connect_open(struct inode *inode, struct file *file)
  52. {
  53. return single_open(file, cnss_pin_connect_show, inode->i_private);
  54. }
  55. static const struct file_operations cnss_pin_connect_fops = {
  56. .read = seq_read,
  57. .release = single_release,
  58. .open = cnss_pin_connect_open,
  59. .owner = THIS_MODULE,
  60. .llseek = seq_lseek,
  61. };
  62. static int cnss_stats_show_state(struct seq_file *s,
  63. struct cnss_plat_data *plat_priv)
  64. {
  65. enum cnss_driver_state i;
  66. int skip = 0;
  67. unsigned long state;
  68. seq_printf(s, "\nState: 0x%lx(", plat_priv->driver_state);
  69. for (i = 0, state = plat_priv->driver_state; state != 0;
  70. state >>= 1, i++) {
  71. if (!(state & 0x1))
  72. continue;
  73. if (skip++)
  74. seq_puts(s, " | ");
  75. switch (i) {
  76. case CNSS_QMI_WLFW_CONNECTED:
  77. seq_puts(s, "QMI_WLFW_CONNECTED");
  78. continue;
  79. case CNSS_FW_MEM_READY:
  80. seq_puts(s, "FW_MEM_READY");
  81. continue;
  82. case CNSS_FW_READY:
  83. seq_puts(s, "FW_READY");
  84. continue;
  85. case CNSS_IN_COLD_BOOT_CAL:
  86. seq_puts(s, "IN_COLD_BOOT_CAL");
  87. continue;
  88. case CNSS_DRIVER_LOADING:
  89. seq_puts(s, "DRIVER_LOADING");
  90. continue;
  91. case CNSS_DRIVER_UNLOADING:
  92. seq_puts(s, "DRIVER_UNLOADING");
  93. continue;
  94. case CNSS_DRIVER_IDLE_RESTART:
  95. seq_puts(s, "IDLE_RESTART");
  96. continue;
  97. case CNSS_DRIVER_IDLE_SHUTDOWN:
  98. seq_puts(s, "IDLE_SHUTDOWN");
  99. continue;
  100. case CNSS_DRIVER_PROBED:
  101. seq_puts(s, "DRIVER_PROBED");
  102. continue;
  103. case CNSS_DRIVER_RECOVERY:
  104. seq_puts(s, "DRIVER_RECOVERY");
  105. continue;
  106. case CNSS_FW_BOOT_RECOVERY:
  107. seq_puts(s, "FW_BOOT_RECOVERY");
  108. continue;
  109. case CNSS_DEV_ERR_NOTIFY:
  110. seq_puts(s, "DEV_ERR");
  111. continue;
  112. case CNSS_DRIVER_DEBUG:
  113. seq_puts(s, "DRIVER_DEBUG");
  114. continue;
  115. case CNSS_COEX_CONNECTED:
  116. seq_puts(s, "COEX_CONNECTED");
  117. continue;
  118. case CNSS_IMS_CONNECTED:
  119. seq_puts(s, "IMS_CONNECTED");
  120. continue;
  121. case CNSS_IN_SUSPEND_RESUME:
  122. seq_puts(s, "IN_SUSPEND_RESUME");
  123. continue;
  124. case CNSS_IN_REBOOT:
  125. seq_puts(s, "IN_REBOOT");
  126. continue;
  127. case CNSS_COLD_BOOT_CAL_DONE:
  128. seq_puts(s, "COLD_BOOT_CAL_DONE");
  129. continue;
  130. case CNSS_IN_PANIC:
  131. seq_puts(s, "IN_PANIC");
  132. continue;
  133. case CNSS_QMI_DEL_SERVER:
  134. seq_puts(s, "DEL_SERVER_IN_PROGRESS");
  135. continue;
  136. case CNSS_QMI_DMS_CONNECTED:
  137. seq_puts(s, "DMS_CONNECTED");
  138. continue;
  139. case CNSS_DMS_DEL_SERVER:
  140. seq_puts(s, "DMS_DEL_SERVER");
  141. continue;
  142. case CNSS_DAEMON_CONNECTED:
  143. seq_puts(s, "DAEMON_CONNECTED");
  144. continue;
  145. case CNSS_PCI_PROBE_DONE:
  146. seq_puts(s, "PCI PROBE DONE");
  147. continue;
  148. case CNSS_DRIVER_REGISTER:
  149. seq_puts(s, "DRIVER REGISTERED");
  150. continue;
  151. case CNSS_WLAN_HW_DISABLED:
  152. seq_puts(s, "WLAN HW DISABLED");
  153. continue;
  154. case CNSS_FS_READY:
  155. seq_puts(s, "FS READY");
  156. continue;
  157. case CNSS_DRIVER_REGISTERED:
  158. seq_puts(s, "DRIVER REGISTERED");
  159. continue;
  160. }
  161. seq_printf(s, "UNKNOWN-%d", i);
  162. }
  163. seq_puts(s, ")\n");
  164. return 0;
  165. }
  166. static int cnss_stats_show_gpio_state(struct seq_file *s,
  167. struct cnss_plat_data *plat_priv)
  168. {
  169. seq_printf(s, "\nHost SOL: %d", cnss_get_host_sol_value(plat_priv));
  170. seq_printf(s, "\nDev SOL: %d", cnss_get_dev_sol_value(plat_priv));
  171. return 0;
  172. }
  173. static int cnss_stats_show(struct seq_file *s, void *data)
  174. {
  175. struct cnss_plat_data *plat_priv = s->private;
  176. cnss_stats_show_state(s, plat_priv);
  177. cnss_stats_show_gpio_state(s, plat_priv);
  178. return 0;
  179. }
  180. static int cnss_stats_open(struct inode *inode, struct file *file)
  181. {
  182. return single_open(file, cnss_stats_show, inode->i_private);
  183. }
  184. static const struct file_operations cnss_stats_fops = {
  185. .read = seq_read,
  186. .release = single_release,
  187. .open = cnss_stats_open,
  188. .owner = THIS_MODULE,
  189. .llseek = seq_lseek,
  190. };
  191. static ssize_t cnss_dev_boot_debug_write(struct file *fp,
  192. const char __user *user_buf,
  193. size_t count, loff_t *off)
  194. {
  195. struct cnss_plat_data *plat_priv =
  196. ((struct seq_file *)fp->private_data)->private;
  197. struct cnss_pci_data *pci_priv;
  198. char buf[64];
  199. char *cmd;
  200. unsigned int len = 0;
  201. char *sptr, *token;
  202. const char *delim = " ";
  203. int ret = 0;
  204. if (!plat_priv)
  205. return -ENODEV;
  206. len = min(count, sizeof(buf) - 1);
  207. if (copy_from_user(buf, user_buf, len))
  208. return -EFAULT;
  209. buf[len] = '\0';
  210. sptr = buf;
  211. token = strsep(&sptr, delim);
  212. if (!token)
  213. return -EINVAL;
  214. cmd = token;
  215. cnss_pr_dbg("Received dev_boot debug command: %s\n", cmd);
  216. if (sysfs_streq(cmd, "on")) {
  217. ret = cnss_power_on_device(plat_priv, false);
  218. } else if (sysfs_streq(cmd, "off")) {
  219. cnss_power_off_device(plat_priv);
  220. } else if (sysfs_streq(cmd, "enumerate")) {
  221. ret = cnss_pci_init(plat_priv);
  222. } else if (sysfs_streq(cmd, "powerup")) {
  223. set_bit(CNSS_DRIVER_DEBUG, &plat_priv->driver_state);
  224. ret = cnss_driver_event_post(plat_priv,
  225. CNSS_DRIVER_EVENT_POWER_UP,
  226. CNSS_EVENT_SYNC, NULL);
  227. } else if (sysfs_streq(cmd, "shutdown")) {
  228. ret = cnss_driver_event_post(plat_priv,
  229. CNSS_DRIVER_EVENT_POWER_DOWN,
  230. 0, NULL);
  231. clear_bit(CNSS_DRIVER_DEBUG, &plat_priv->driver_state);
  232. } else if (sysfs_streq(cmd, "assert_host_sol")) {
  233. ret = cnss_set_host_sol_value(plat_priv, 1);
  234. } else if (sysfs_streq(cmd, "deassert_host_sol")) {
  235. ret = cnss_set_host_sol_value(plat_priv, 0);
  236. } else if (sysfs_streq(cmd, "pdc_update")) {
  237. if (!sptr)
  238. return -EINVAL;
  239. ret = cnss_aop_send_msg(plat_priv, sptr);
  240. } else if (sysfs_streq(cmd, "dev_check")) {
  241. cnss_wlan_hw_disable_check(plat_priv);
  242. } else if (sysfs_streq(cmd, "dev_enable")) {
  243. cnss_wlan_hw_enable();
  244. } else {
  245. pci_priv = plat_priv->bus_priv;
  246. if (!pci_priv)
  247. return -ENODEV;
  248. if (sysfs_streq(cmd, "download")) {
  249. set_bit(CNSS_DRIVER_DEBUG, &plat_priv->driver_state);
  250. ret = cnss_pci_start_mhi(pci_priv);
  251. } else if (sysfs_streq(cmd, "linkup")) {
  252. ret = cnss_resume_pci_link(pci_priv);
  253. } else if (sysfs_streq(cmd, "linkdown")) {
  254. ret = cnss_suspend_pci_link(pci_priv);
  255. } else if (sysfs_streq(cmd, "assert")) {
  256. cnss_pr_info("FW Assert triggered for debug\n");
  257. ret = cnss_force_fw_assert(&pci_priv->pci_dev->dev);
  258. } else if (sysfs_streq(cmd, "set_cbc_done")) {
  259. cnss_pr_dbg("Force set cold boot cal done status\n");
  260. set_bit(CNSS_COLD_BOOT_CAL_DONE,
  261. &plat_priv->driver_state);
  262. } else {
  263. cnss_pr_err("Device boot debugfs command is invalid\n");
  264. ret = -EINVAL;
  265. }
  266. }
  267. if (ret < 0)
  268. return ret;
  269. return count;
  270. }
  271. static int cnss_dev_boot_debug_show(struct seq_file *s, void *data)
  272. {
  273. seq_puts(s, "\nUsage: echo <action> > <debugfs_path>/cnss/dev_boot\n");
  274. seq_puts(s, "<action> can be one of below:\n");
  275. seq_puts(s, "on: turn on device power, assert WLAN_EN\n");
  276. seq_puts(s, "off: de-assert WLAN_EN, turn off device power\n");
  277. seq_puts(s, "enumerate: de-assert PERST, enumerate PCIe\n");
  278. seq_puts(s, "download: download FW and do QMI handshake with FW\n");
  279. seq_puts(s, "linkup: bring up PCIe link\n");
  280. seq_puts(s, "linkdown: bring down PCIe link\n");
  281. seq_puts(s, "powerup: full power on sequence to boot device, download FW and do QMI handshake with FW\n");
  282. seq_puts(s, "shutdown: full power off sequence to shutdown device\n");
  283. seq_puts(s, "assert: trigger firmware assert\n");
  284. seq_puts(s, "set_cbc_done: Set cold boot calibration done status\n");
  285. seq_puts(s, "\npdc_update usage:");
  286. seq_puts(s, "1. echo pdc_update {class: wlan_pdc ss: <pdc_ss>, res: <vreg>.<mode>, <seq>: <val>} > <debugfs_path>/cnss/dev_boot\n");
  287. seq_puts(s, "2. echo pdc_update {class: wlan_pdc ss: <pdc_ss>, res: pdc, enable: <val>} > <debugfs_path>/cnss/dev_boot\n");
  288. return 0;
  289. }
  290. static int cnss_dev_boot_debug_open(struct inode *inode, struct file *file)
  291. {
  292. return single_open(file, cnss_dev_boot_debug_show, inode->i_private);
  293. }
  294. static const struct file_operations cnss_dev_boot_debug_fops = {
  295. .read = seq_read,
  296. .write = cnss_dev_boot_debug_write,
  297. .release = single_release,
  298. .open = cnss_dev_boot_debug_open,
  299. .owner = THIS_MODULE,
  300. .llseek = seq_lseek,
  301. };
  302. static int cnss_reg_read_debug_show(struct seq_file *s, void *data)
  303. {
  304. struct cnss_plat_data *plat_priv = s->private;
  305. mutex_lock(&plat_priv->dev_lock);
  306. if (!plat_priv->diag_reg_read_buf) {
  307. seq_puts(s, "\nUsage: echo <mem_type> <offset> <data_len> > <debugfs_path>/cnss/reg_read\n");
  308. seq_puts(s, "Use mem_type = 0xff for register read by IO access, data_len will be ignored\n");
  309. seq_puts(s, "Use mem_type = 0xfe for register read by raw IO access which skips sanity checks, data_len will be ignored\n");
  310. seq_puts(s, "Use other mem_type for register read by QMI\n");
  311. mutex_unlock(&plat_priv->dev_lock);
  312. return 0;
  313. }
  314. seq_printf(s, "\nRegister read, address: 0x%x memory type: 0x%x length: 0x%x\n\n",
  315. plat_priv->diag_reg_read_addr,
  316. plat_priv->diag_reg_read_mem_type,
  317. plat_priv->diag_reg_read_len);
  318. seq_hex_dump(s, "", DUMP_PREFIX_OFFSET, 32, 4,
  319. plat_priv->diag_reg_read_buf,
  320. plat_priv->diag_reg_read_len, false);
  321. plat_priv->diag_reg_read_len = 0;
  322. kfree(plat_priv->diag_reg_read_buf);
  323. plat_priv->diag_reg_read_buf = NULL;
  324. mutex_unlock(&plat_priv->dev_lock);
  325. return 0;
  326. }
  327. static ssize_t cnss_reg_read_debug_write(struct file *fp,
  328. const char __user *user_buf,
  329. size_t count, loff_t *off)
  330. {
  331. struct cnss_plat_data *plat_priv =
  332. ((struct seq_file *)fp->private_data)->private;
  333. char buf[64];
  334. char *sptr, *token;
  335. unsigned int len = 0;
  336. u32 reg_offset, mem_type;
  337. u32 data_len = 0, reg_val = 0;
  338. u8 *reg_buf = NULL;
  339. const char *delim = " ";
  340. int ret = 0;
  341. len = min(count, sizeof(buf) - 1);
  342. if (copy_from_user(buf, user_buf, len))
  343. return -EFAULT;
  344. buf[len] = '\0';
  345. sptr = buf;
  346. token = strsep(&sptr, delim);
  347. if (!token)
  348. return -EINVAL;
  349. if (!sptr)
  350. return -EINVAL;
  351. if (kstrtou32(token, 0, &mem_type))
  352. return -EINVAL;
  353. token = strsep(&sptr, delim);
  354. if (!token)
  355. return -EINVAL;
  356. if (!sptr)
  357. return -EINVAL;
  358. if (kstrtou32(token, 0, &reg_offset))
  359. return -EINVAL;
  360. token = strsep(&sptr, delim);
  361. if (!token)
  362. return -EINVAL;
  363. if (kstrtou32(token, 0, &data_len))
  364. return -EINVAL;
  365. if (mem_type == MMIO_REG_ACCESS_MEM_TYPE ||
  366. mem_type == MMIO_REG_RAW_ACCESS_MEM_TYPE) {
  367. ret = cnss_bus_debug_reg_read(plat_priv, reg_offset, &reg_val,
  368. mem_type ==
  369. MMIO_REG_RAW_ACCESS_MEM_TYPE);
  370. if (ret)
  371. return ret;
  372. cnss_pr_dbg("Read 0x%x from register offset 0x%x\n", reg_val,
  373. reg_offset);
  374. return count;
  375. }
  376. if (!test_bit(CNSS_FW_READY, &plat_priv->driver_state)) {
  377. cnss_pr_err("Firmware is not ready yet\n");
  378. return -EINVAL;
  379. }
  380. mutex_lock(&plat_priv->dev_lock);
  381. kfree(plat_priv->diag_reg_read_buf);
  382. plat_priv->diag_reg_read_buf = NULL;
  383. reg_buf = kzalloc(data_len, GFP_KERNEL);
  384. if (!reg_buf) {
  385. mutex_unlock(&plat_priv->dev_lock);
  386. return -ENOMEM;
  387. }
  388. ret = cnss_wlfw_athdiag_read_send_sync(plat_priv, reg_offset,
  389. mem_type, data_len,
  390. reg_buf);
  391. if (ret) {
  392. kfree(reg_buf);
  393. mutex_unlock(&plat_priv->dev_lock);
  394. return ret;
  395. }
  396. plat_priv->diag_reg_read_addr = reg_offset;
  397. plat_priv->diag_reg_read_mem_type = mem_type;
  398. plat_priv->diag_reg_read_len = data_len;
  399. plat_priv->diag_reg_read_buf = reg_buf;
  400. mutex_unlock(&plat_priv->dev_lock);
  401. return count;
  402. }
  403. static int cnss_reg_read_debug_open(struct inode *inode, struct file *file)
  404. {
  405. return single_open(file, cnss_reg_read_debug_show, inode->i_private);
  406. }
  407. static const struct file_operations cnss_reg_read_debug_fops = {
  408. .read = seq_read,
  409. .write = cnss_reg_read_debug_write,
  410. .open = cnss_reg_read_debug_open,
  411. .owner = THIS_MODULE,
  412. .llseek = seq_lseek,
  413. };
  414. static int cnss_reg_write_debug_show(struct seq_file *s, void *data)
  415. {
  416. seq_puts(s, "\nUsage: echo <mem_type> <offset> <reg_val> > <debugfs_path>/cnss/reg_write\n");
  417. seq_puts(s, "Use mem_type = 0xff for register write by IO access\n");
  418. seq_puts(s, "Use mem_type = 0xfe for register write by raw IO access which skips sanity checks\n");
  419. seq_puts(s, "Use other mem_type for register write by QMI\n");
  420. return 0;
  421. }
  422. static ssize_t cnss_reg_write_debug_write(struct file *fp,
  423. const char __user *user_buf,
  424. size_t count, loff_t *off)
  425. {
  426. struct cnss_plat_data *plat_priv =
  427. ((struct seq_file *)fp->private_data)->private;
  428. char buf[64];
  429. char *sptr, *token;
  430. unsigned int len = 0;
  431. u32 reg_offset, mem_type, reg_val;
  432. const char *delim = " ";
  433. int ret = 0;
  434. len = min(count, sizeof(buf) - 1);
  435. if (copy_from_user(buf, user_buf, len))
  436. return -EFAULT;
  437. buf[len] = '\0';
  438. sptr = buf;
  439. token = strsep(&sptr, delim);
  440. if (!token)
  441. return -EINVAL;
  442. if (!sptr)
  443. return -EINVAL;
  444. if (kstrtou32(token, 0, &mem_type))
  445. return -EINVAL;
  446. token = strsep(&sptr, delim);
  447. if (!token)
  448. return -EINVAL;
  449. if (!sptr)
  450. return -EINVAL;
  451. if (kstrtou32(token, 0, &reg_offset))
  452. return -EINVAL;
  453. token = strsep(&sptr, delim);
  454. if (!token)
  455. return -EINVAL;
  456. if (kstrtou32(token, 0, &reg_val))
  457. return -EINVAL;
  458. if (mem_type == MMIO_REG_ACCESS_MEM_TYPE ||
  459. mem_type == MMIO_REG_RAW_ACCESS_MEM_TYPE) {
  460. ret = cnss_bus_debug_reg_write(plat_priv, reg_offset, reg_val,
  461. mem_type ==
  462. MMIO_REG_RAW_ACCESS_MEM_TYPE);
  463. if (ret)
  464. return ret;
  465. cnss_pr_dbg("Wrote 0x%x to register offset 0x%x\n", reg_val,
  466. reg_offset);
  467. return count;
  468. }
  469. if (!test_bit(CNSS_FW_READY, &plat_priv->driver_state)) {
  470. cnss_pr_err("Firmware is not ready yet\n");
  471. return -EINVAL;
  472. }
  473. ret = cnss_wlfw_athdiag_write_send_sync(plat_priv, reg_offset, mem_type,
  474. sizeof(u32),
  475. (u8 *)&reg_val);
  476. if (ret)
  477. return ret;
  478. return count;
  479. }
  480. static int cnss_reg_write_debug_open(struct inode *inode, struct file *file)
  481. {
  482. return single_open(file, cnss_reg_write_debug_show, inode->i_private);
  483. }
  484. static const struct file_operations cnss_reg_write_debug_fops = {
  485. .read = seq_read,
  486. .write = cnss_reg_write_debug_write,
  487. .open = cnss_reg_write_debug_open,
  488. .owner = THIS_MODULE,
  489. .llseek = seq_lseek,
  490. };
  491. static ssize_t cnss_runtime_pm_debug_write(struct file *fp,
  492. const char __user *user_buf,
  493. size_t count, loff_t *off)
  494. {
  495. struct cnss_plat_data *plat_priv =
  496. ((struct seq_file *)fp->private_data)->private;
  497. struct cnss_pci_data *pci_priv;
  498. char buf[64];
  499. char *cmd;
  500. unsigned int len = 0;
  501. int ret = 0;
  502. if (!plat_priv)
  503. return -ENODEV;
  504. pci_priv = plat_priv->bus_priv;
  505. if (!pci_priv)
  506. return -ENODEV;
  507. len = min(count, sizeof(buf) - 1);
  508. if (copy_from_user(buf, user_buf, len))
  509. return -EFAULT;
  510. buf[len] = '\0';
  511. cmd = buf;
  512. cnss_pr_dbg("Received runtime_pm debug command: %s\n", cmd);
  513. if (sysfs_streq(cmd, "usage_count")) {
  514. cnss_pci_pm_runtime_show_usage_count(pci_priv);
  515. } else if (sysfs_streq(cmd, "request_resume")) {
  516. ret = cnss_pci_pm_request_resume(pci_priv);
  517. } else if (sysfs_streq(cmd, "resume")) {
  518. ret = cnss_pci_pm_runtime_resume(pci_priv);
  519. } else if (sysfs_streq(cmd, "get")) {
  520. ret = cnss_pci_pm_runtime_get(pci_priv, RTPM_ID_CNSS);
  521. } else if (sysfs_streq(cmd, "get_noresume")) {
  522. cnss_pci_pm_runtime_get_noresume(pci_priv, RTPM_ID_CNSS);
  523. } else if (sysfs_streq(cmd, "put_autosuspend")) {
  524. ret = cnss_pci_pm_runtime_put_autosuspend(pci_priv,
  525. RTPM_ID_CNSS);
  526. } else if (sysfs_streq(cmd, "put_noidle")) {
  527. cnss_pci_pm_runtime_put_noidle(pci_priv, RTPM_ID_CNSS);
  528. } else if (sysfs_streq(cmd, "mark_last_busy")) {
  529. cnss_pci_pm_runtime_mark_last_busy(pci_priv);
  530. } else if (sysfs_streq(cmd, "resume_bus")) {
  531. cnss_pci_resume_bus(pci_priv);
  532. } else if (sysfs_streq(cmd, "suspend_bus")) {
  533. cnss_pci_suspend_bus(pci_priv);
  534. } else {
  535. cnss_pr_err("Runtime PM debugfs command is invalid\n");
  536. ret = -EINVAL;
  537. }
  538. if (ret < 0)
  539. return ret;
  540. return count;
  541. }
  542. static int cnss_runtime_pm_debug_show(struct seq_file *s, void *data)
  543. {
  544. struct cnss_plat_data *plat_priv = s->private;
  545. struct cnss_pci_data *pci_priv;
  546. int i;
  547. if (!plat_priv)
  548. return -ENODEV;
  549. pci_priv = plat_priv->bus_priv;
  550. if (!pci_priv)
  551. return -ENODEV;
  552. seq_puts(s, "\nUsage: echo <action> > <debugfs_path>/cnss/runtime_pm\n");
  553. seq_puts(s, "<action> can be one of below:\n");
  554. seq_puts(s, "usage_count: get runtime PM usage count\n");
  555. seq_puts(s, "reques_resume: do async runtime PM resume\n");
  556. seq_puts(s, "resume: do sync runtime PM resume\n");
  557. seq_puts(s, "get: do runtime PM get\n");
  558. seq_puts(s, "get_noresume: do runtime PM get noresume\n");
  559. seq_puts(s, "put_noidle: do runtime PM put noidle\n");
  560. seq_puts(s, "put_autosuspend: do runtime PM put autosuspend\n");
  561. seq_puts(s, "mark_last_busy: do runtime PM mark last busy\n");
  562. seq_puts(s, "resume_bus: do bus resume only\n");
  563. seq_puts(s, "suspend_bus: do bus suspend only\n");
  564. seq_puts(s, "\nStats:\n");
  565. seq_printf(s, "%s: %u\n", "get count",
  566. atomic_read(&pci_priv->pm_stats.runtime_get));
  567. seq_printf(s, "%s: %u\n", "put count",
  568. atomic_read(&pci_priv->pm_stats.runtime_put));
  569. seq_printf(s, "%-10s%-10s%-10s%-15s%-15s\n",
  570. "id:", "get", "put", "get time(us)", "put time(us)");
  571. for (i = 0; i < RTPM_ID_MAX; i++) {
  572. seq_printf(s, "%d%-9s", i, ":");
  573. seq_printf(s, "%-10d",
  574. atomic_read(&pci_priv->pm_stats.runtime_get_id[i]));
  575. seq_printf(s, "%-10d",
  576. atomic_read(&pci_priv->pm_stats.runtime_put_id[i]));
  577. seq_printf(s, "%-15llu",
  578. pci_priv->pm_stats.runtime_get_timestamp_id[i]);
  579. seq_printf(s, "%-15llu\n",
  580. pci_priv->pm_stats.runtime_put_timestamp_id[i]);
  581. }
  582. return 0;
  583. }
  584. static int cnss_runtime_pm_debug_open(struct inode *inode, struct file *file)
  585. {
  586. return single_open(file, cnss_runtime_pm_debug_show, inode->i_private);
  587. }
  588. static const struct file_operations cnss_runtime_pm_debug_fops = {
  589. .read = seq_read,
  590. .write = cnss_runtime_pm_debug_write,
  591. .open = cnss_runtime_pm_debug_open,
  592. .owner = THIS_MODULE,
  593. .llseek = seq_lseek,
  594. };
  595. static int process_drv(struct cnss_plat_data *plat_priv, bool enabled)
  596. {
  597. if (test_bit(CNSS_QMI_WLFW_CONNECTED, &plat_priv->driver_state)) {
  598. cnss_pr_err("DRV cmd must be used before QMI ready\n");
  599. return -EINVAL;
  600. }
  601. enabled ? cnss_set_feature_list(plat_priv, CNSS_DRV_SUPPORT_V01) :
  602. cnss_clear_feature_list(plat_priv, CNSS_DRV_SUPPORT_V01);
  603. cnss_pr_info("%s DRV suspend\n", enabled ? "enable" : "disable");
  604. return 0;
  605. }
  606. static int process_quirks(struct cnss_plat_data *plat_priv, u32 val)
  607. {
  608. enum cnss_debug_quirks i;
  609. int ret = 0;
  610. unsigned long state;
  611. unsigned long quirks = 0;
  612. for (i = 0, state = val; i < QUIRK_MAX_VALUE; state >>= 1, i++) {
  613. switch (i) {
  614. case DISABLE_DRV:
  615. ret = process_drv(plat_priv, !(state & 0x1));
  616. if (!ret)
  617. quirks |= (state & 0x1) << i;
  618. continue;
  619. default:
  620. quirks |= (state & 0x1) << i;
  621. continue;
  622. }
  623. }
  624. plat_priv->ctrl_params.quirks = quirks;
  625. return 0;
  626. }
  627. static ssize_t cnss_control_params_debug_write(struct file *fp,
  628. const char __user *user_buf,
  629. size_t count, loff_t *off)
  630. {
  631. struct cnss_plat_data *plat_priv =
  632. ((struct seq_file *)fp->private_data)->private;
  633. char buf[64];
  634. char *sptr, *token;
  635. char *cmd;
  636. u32 val;
  637. unsigned int len = 0;
  638. const char *delim = " ";
  639. if (!plat_priv)
  640. return -ENODEV;
  641. len = min(count, sizeof(buf) - 1);
  642. if (copy_from_user(buf, user_buf, len))
  643. return -EFAULT;
  644. buf[len] = '\0';
  645. sptr = buf;
  646. token = strsep(&sptr, delim);
  647. if (!token)
  648. return -EINVAL;
  649. if (!sptr)
  650. return -EINVAL;
  651. cmd = token;
  652. token = strsep(&sptr, delim);
  653. if (!token)
  654. return -EINVAL;
  655. if (kstrtou32(token, 0, &val))
  656. return -EINVAL;
  657. if (strcmp(cmd, "quirks") == 0)
  658. process_quirks(plat_priv, val);
  659. else if (strcmp(cmd, "mhi_timeout") == 0)
  660. plat_priv->ctrl_params.mhi_timeout = val;
  661. else if (strcmp(cmd, "mhi_m2_timeout") == 0)
  662. plat_priv->ctrl_params.mhi_m2_timeout = val;
  663. else if (strcmp(cmd, "qmi_timeout") == 0)
  664. plat_priv->ctrl_params.qmi_timeout = val;
  665. else if (strcmp(cmd, "bdf_type") == 0)
  666. plat_priv->ctrl_params.bdf_type = val;
  667. else if (strcmp(cmd, "time_sync_period") == 0)
  668. plat_priv->ctrl_params.time_sync_period = val;
  669. else if (strcmp(cmd, "kern_log_level") == 0) {
  670. if (val < MAX_LOG)
  671. cnss_kernel_log_level = val;
  672. } else if (strcmp(cmd, "ipc_log_level") == 0) {
  673. return cnss_set_ipc_log_level(val) ? -EINVAL : count;
  674. } else
  675. return -EINVAL;
  676. return count;
  677. }
  678. static int cnss_show_quirks_state(struct seq_file *s,
  679. struct cnss_plat_data *plat_priv)
  680. {
  681. enum cnss_debug_quirks i;
  682. int skip = 0;
  683. unsigned long state;
  684. seq_printf(s, "quirks: 0x%lx (", plat_priv->ctrl_params.quirks);
  685. for (i = 0, state = plat_priv->ctrl_params.quirks;
  686. state != 0; state >>= 1, i++) {
  687. if (!(state & 0x1))
  688. continue;
  689. if (skip++)
  690. seq_puts(s, " | ");
  691. switch (i) {
  692. case LINK_DOWN_SELF_RECOVERY:
  693. seq_puts(s, "LINK_DOWN_SELF_RECOVERY");
  694. continue;
  695. case SKIP_DEVICE_BOOT:
  696. seq_puts(s, "SKIP_DEVICE_BOOT");
  697. continue;
  698. case USE_CORE_ONLY_FW:
  699. seq_puts(s, "USE_CORE_ONLY_FW");
  700. continue;
  701. case SKIP_RECOVERY:
  702. seq_puts(s, "SKIP_RECOVERY");
  703. continue;
  704. case QMI_BYPASS:
  705. seq_puts(s, "QMI_BYPASS");
  706. continue;
  707. case ENABLE_WALTEST:
  708. seq_puts(s, "WALTEST");
  709. continue;
  710. case ENABLE_PCI_LINK_DOWN_PANIC:
  711. seq_puts(s, "PCI_LINK_DOWN_PANIC");
  712. continue;
  713. case FBC_BYPASS:
  714. seq_puts(s, "FBC_BYPASS");
  715. continue;
  716. case ENABLE_DAEMON_SUPPORT:
  717. seq_puts(s, "DAEMON_SUPPORT");
  718. continue;
  719. case DISABLE_DRV:
  720. seq_puts(s, "DISABLE_DRV");
  721. continue;
  722. case DISABLE_IO_COHERENCY:
  723. seq_puts(s, "DISABLE_IO_COHERENCY");
  724. continue;
  725. case IGNORE_PCI_LINK_FAILURE:
  726. seq_puts(s, "IGNORE_PCI_LINK_FAILURE");
  727. continue;
  728. case DISABLE_TIME_SYNC:
  729. seq_puts(s, "DISABLE_TIME_SYNC");
  730. continue;
  731. case FORCE_ONE_MSI:
  732. seq_puts(s, "FORCE_ONE_MSI");
  733. continue;
  734. default:
  735. continue;
  736. }
  737. }
  738. seq_puts(s, ")\n");
  739. return 0;
  740. }
  741. static int cnss_control_params_debug_show(struct seq_file *s, void *data)
  742. {
  743. struct cnss_plat_data *cnss_priv = s->private;
  744. u32 ipc_log_level;
  745. seq_puts(s, "\nUsage: echo <params_name> <value> > <debugfs_path>/cnss/control_params\n");
  746. seq_puts(s, "<params_name> can be one of below:\n");
  747. seq_puts(s, "quirks: Debug quirks for driver\n");
  748. seq_puts(s, "mhi_timeout: Timeout for MHI operation in milliseconds\n");
  749. seq_puts(s, "qmi_timeout: Timeout for QMI message in milliseconds\n");
  750. seq_puts(s, "bdf_type: Type of board data file to be downloaded\n");
  751. seq_puts(s, "time_sync_period: Time period to do time sync with device in milliseconds\n");
  752. seq_puts(s, "\nCurrent value:\n");
  753. cnss_show_quirks_state(s, cnss_priv);
  754. seq_printf(s, "mhi_timeout: %u\n", cnss_priv->ctrl_params.mhi_timeout);
  755. seq_printf(s, "mhi_m2_timeout: %u\n",
  756. cnss_priv->ctrl_params.mhi_m2_timeout);
  757. seq_printf(s, "qmi_timeout: %u\n", cnss_priv->ctrl_params.qmi_timeout);
  758. seq_printf(s, "bdf_type: %u\n", cnss_priv->ctrl_params.bdf_type);
  759. seq_printf(s, "time_sync_period: %u\n",
  760. cnss_priv->ctrl_params.time_sync_period);
  761. seq_printf(s, "kern_log_level: %u\n", cnss_kernel_log_level);
  762. ipc_log_level = cnss_get_ipc_log_level();
  763. if (ipc_log_level != MAX_LOG)
  764. seq_printf(s, "ipc_log_level: %u\n", ipc_log_level);
  765. return 0;
  766. }
  767. static int cnss_control_params_debug_open(struct inode *inode,
  768. struct file *file)
  769. {
  770. return single_open(file, cnss_control_params_debug_show,
  771. inode->i_private);
  772. }
  773. static const struct file_operations cnss_control_params_debug_fops = {
  774. .read = seq_read,
  775. .write = cnss_control_params_debug_write,
  776. .open = cnss_control_params_debug_open,
  777. .owner = THIS_MODULE,
  778. .llseek = seq_lseek,
  779. };
  780. static ssize_t cnss_dynamic_feature_write(struct file *fp,
  781. const char __user *user_buf,
  782. size_t count, loff_t *off)
  783. {
  784. struct cnss_plat_data *plat_priv =
  785. ((struct seq_file *)fp->private_data)->private;
  786. int ret = 0;
  787. u64 val;
  788. ret = kstrtou64_from_user(user_buf, count, 0, &val);
  789. if (ret)
  790. return ret;
  791. plat_priv->dynamic_feature = val;
  792. ret = cnss_wlfw_dynamic_feature_mask_send_sync(plat_priv);
  793. if (ret < 0)
  794. return ret;
  795. return count;
  796. }
  797. static int cnss_dynamic_feature_show(struct seq_file *s, void *data)
  798. {
  799. struct cnss_plat_data *cnss_priv = s->private;
  800. seq_printf(s, "dynamic_feature: 0x%llx\n", cnss_priv->dynamic_feature);
  801. return 0;
  802. }
  803. static int cnss_dynamic_feature_open(struct inode *inode,
  804. struct file *file)
  805. {
  806. return single_open(file, cnss_dynamic_feature_show,
  807. inode->i_private);
  808. }
  809. static const struct file_operations cnss_dynamic_feature_fops = {
  810. .read = seq_read,
  811. .write = cnss_dynamic_feature_write,
  812. .open = cnss_dynamic_feature_open,
  813. .owner = THIS_MODULE,
  814. .llseek = seq_lseek,
  815. };
  816. static int cnss_smmu_fault_timestamp_show(struct seq_file *s, void *data)
  817. {
  818. struct cnss_plat_data *plat_priv = s->private;
  819. struct cnss_pci_data *pci_priv = plat_priv->bus_priv;
  820. if (!pci_priv)
  821. return -ENODEV;
  822. seq_printf(s, "smmu irq cb entry timestamp : %llu ns\n",
  823. pci_priv->smmu_fault_timestamp[SMMU_CB_ENTRY]);
  824. seq_printf(s, "smmu irq cb before doorbell ring timestamp : %llu ns\n",
  825. pci_priv->smmu_fault_timestamp[SMMU_CB_DOORBELL_RING]);
  826. seq_printf(s, "smmu irq cb after doorbell ring timestamp : %llu ns\n",
  827. pci_priv->smmu_fault_timestamp[SMMU_CB_EXIT]);
  828. return 0;
  829. }
  830. static int cnss_smmu_fault_timestamp_open(struct inode *inode,
  831. struct file *file)
  832. {
  833. return single_open(file, cnss_smmu_fault_timestamp_show,
  834. inode->i_private);
  835. }
  836. static const struct file_operations cnss_smmu_fault_timestamp_fops = {
  837. .read = seq_read,
  838. .release = single_release,
  839. .open = cnss_smmu_fault_timestamp_open,
  840. .owner = THIS_MODULE,
  841. .llseek = seq_lseek,
  842. };
  843. #ifdef CONFIG_DEBUG_FS
  844. #ifdef CONFIG_CNSS2_DEBUG
  845. static int cnss_create_debug_only_node(struct cnss_plat_data *plat_priv)
  846. {
  847. struct dentry *root_dentry = plat_priv->root_dentry;
  848. debugfs_create_file("dev_boot", 0600, root_dentry, plat_priv,
  849. &cnss_dev_boot_debug_fops);
  850. debugfs_create_file("reg_read", 0600, root_dentry, plat_priv,
  851. &cnss_reg_read_debug_fops);
  852. debugfs_create_file("reg_write", 0600, root_dentry, plat_priv,
  853. &cnss_reg_write_debug_fops);
  854. debugfs_create_file("runtime_pm", 0600, root_dentry, plat_priv,
  855. &cnss_runtime_pm_debug_fops);
  856. debugfs_create_file("control_params", 0600, root_dentry, plat_priv,
  857. &cnss_control_params_debug_fops);
  858. debugfs_create_file("dynamic_feature", 0600, root_dentry, plat_priv,
  859. &cnss_dynamic_feature_fops);
  860. debugfs_create_file("cnss_smmu_fault_timestamp", 0600, root_dentry,
  861. plat_priv, &cnss_smmu_fault_timestamp_fops);
  862. return 0;
  863. }
  864. #else
  865. static int cnss_create_debug_only_node(struct cnss_plat_data *plat_priv)
  866. {
  867. return 0;
  868. }
  869. #endif
  870. int cnss_debugfs_create(struct cnss_plat_data *plat_priv)
  871. {
  872. int ret = 0;
  873. struct dentry *root_dentry;
  874. char name[CNSS_FS_NAME_SIZE];
  875. if (cnss_is_dual_wlan_enabled())
  876. snprintf(name, CNSS_FS_NAME_SIZE, CNSS_FS_NAME "_%d",
  877. plat_priv->plat_idx);
  878. else
  879. snprintf(name, CNSS_FS_NAME_SIZE, CNSS_FS_NAME);
  880. root_dentry = debugfs_create_dir(name, 0);
  881. if (IS_ERR(root_dentry)) {
  882. ret = PTR_ERR(root_dentry);
  883. cnss_pr_err("Unable to create debugfs %d\n", ret);
  884. goto out;
  885. }
  886. plat_priv->root_dentry = root_dentry;
  887. debugfs_create_file("pin_connect_result", 0644, root_dentry, plat_priv,
  888. &cnss_pin_connect_fops);
  889. debugfs_create_file("stats", 0644, root_dentry, plat_priv,
  890. &cnss_stats_fops);
  891. cnss_create_debug_only_node(plat_priv);
  892. out:
  893. return ret;
  894. }
  895. void cnss_debugfs_destroy(struct cnss_plat_data *plat_priv)
  896. {
  897. debugfs_remove_recursive(plat_priv->root_dentry);
  898. }
  899. #else
  900. int cnss_debugfs_create(struct cnss_plat_data *plat_priv)
  901. {
  902. plat_priv->root_dentry = NULL;
  903. return 0;
  904. }
  905. void cnss_debugfs_destroy(struct cnss_plat_data *plat_priv)
  906. {
  907. }
  908. #endif
  909. #if IS_ENABLED(CONFIG_IPC_LOGGING)
  910. void cnss_debug_ipc_log_print(void *log_ctx, char *process, const char *fn,
  911. enum log_level kern_log_level,
  912. enum log_level ipc_log_level, char *fmt, ...)
  913. {
  914. struct va_format vaf;
  915. va_list va_args;
  916. va_start(va_args, fmt);
  917. vaf.fmt = fmt;
  918. vaf.va = &va_args;
  919. if (kern_log_level <= cnss_kernel_log_level) {
  920. switch (kern_log_level) {
  921. case EMERG_LOG:
  922. pr_emerg("cnss: %pV", &vaf);
  923. break;
  924. case ALERT_LOG:
  925. pr_alert("cnss: %pV", &vaf);
  926. break;
  927. case CRIT_LOG:
  928. pr_crit("cnss: %pV", &vaf);
  929. break;
  930. case ERR_LOG:
  931. pr_err("cnss: %pV", &vaf);
  932. break;
  933. case WARNING_LOG:
  934. pr_warn("cnss: %pV", &vaf);
  935. break;
  936. case NOTICE_LOG:
  937. pr_notice("cnss: %pV", &vaf);
  938. break;
  939. case INFO_LOG:
  940. pr_info("cnss: %pV", &vaf);
  941. break;
  942. case DEBUG_LOG:
  943. case DEBUG_HI_LOG:
  944. pr_debug("cnss: %pV", &vaf);
  945. break;
  946. default:
  947. break;
  948. }
  949. }
  950. if (ipc_log_level <= cnss_ipc_log_level)
  951. ipc_log_string(log_ctx, "[%s] %s: %pV", process, fn, &vaf);
  952. va_end(va_args);
  953. }
  954. static int cnss_ipc_logging_init(void)
  955. {
  956. cnss_ipc_log_context = ipc_log_context_create(CNSS_IPC_LOG_PAGES,
  957. "cnss", 0);
  958. if (!cnss_ipc_log_context) {
  959. cnss_pr_err("Unable to create IPC log context\n");
  960. return -EINVAL;
  961. }
  962. cnss_ipc_log_long_context = ipc_log_context_create(CNSS_IPC_LOG_PAGES,
  963. "cnss-long", 0);
  964. if (!cnss_ipc_log_long_context) {
  965. cnss_pr_err("Unable to create IPC long log context\n");
  966. ipc_log_context_destroy(cnss_ipc_log_context);
  967. return -EINVAL;
  968. }
  969. return 0;
  970. }
  971. static void cnss_ipc_logging_deinit(void)
  972. {
  973. if (cnss_ipc_log_long_context) {
  974. ipc_log_context_destroy(cnss_ipc_log_long_context);
  975. cnss_ipc_log_long_context = NULL;
  976. }
  977. if (cnss_ipc_log_context) {
  978. ipc_log_context_destroy(cnss_ipc_log_context);
  979. cnss_ipc_log_context = NULL;
  980. }
  981. }
  982. #else
  983. static int cnss_ipc_logging_init(void) { return 0; }
  984. static void cnss_ipc_logging_deinit(void) {}
  985. void cnss_debug_ipc_log_print(void *log_ctx, char *process, const char *fn,
  986. enum log_level kern_log_level,
  987. enum log_level ipc_log_level, char *fmt, ...)
  988. {
  989. struct va_format vaf;
  990. va_list va_args;
  991. va_start(va_args, fmt);
  992. vaf.fmt = fmt;
  993. vaf.va = &va_args;
  994. if (kern_log_level <= cnss_kernel_log_level) {
  995. switch (kern_log_level) {
  996. case EMERG_LOG:
  997. pr_emerg("cnss: %pV", &vaf);
  998. break;
  999. case ALERT_LOG:
  1000. pr_alert("cnss: %pV", &vaf);
  1001. break;
  1002. case CRIT_LOG:
  1003. pr_crit("cnss: %pV", &vaf);
  1004. break;
  1005. case ERR_LOG:
  1006. pr_err("cnss: %pV", &vaf);
  1007. break;
  1008. case WARNING_LOG:
  1009. pr_warn("cnss: %pV", &vaf);
  1010. break;
  1011. case NOTICE_LOG:
  1012. pr_notice("cnss: %pV", &vaf);
  1013. break;
  1014. case INFO_LOG:
  1015. pr_info("cnss: %pV", &vaf);
  1016. break;
  1017. case DEBUG_LOG:
  1018. case DEBUG_HI_LOG:
  1019. pr_debug("cnss: %pV", &vaf);
  1020. break;
  1021. default:
  1022. break;
  1023. }
  1024. }
  1025. va_end(va_args);
  1026. }
  1027. #endif
  1028. int cnss_debug_init(void)
  1029. {
  1030. return cnss_ipc_logging_init();
  1031. }
  1032. void cnss_debug_deinit(void)
  1033. {
  1034. cnss_ipc_logging_deinit();
  1035. }