iosm_ipc_trace.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) 2020-2021 Intel Corporation.
  4. */
  5. #include <linux/wwan.h>
  6. #include "iosm_ipc_trace.h"
  7. /* sub buffer size and number of sub buffer */
  8. #define IOSM_TRC_SUB_BUFF_SIZE 131072
  9. #define IOSM_TRC_N_SUB_BUFF 32
  10. #define IOSM_TRC_FILE_PERM 0600
  11. #define IOSM_TRC_DEBUGFS_TRACE "trace"
  12. #define IOSM_TRC_DEBUGFS_TRACE_CTRL "trace_ctrl"
  13. /**
  14. * ipc_trace_port_rx - Receive trace packet from cp and write to relay buffer
  15. * @ipc_imem: Pointer to iosm_imem structure
  16. * @skb: Pointer to struct sk_buff
  17. */
  18. void ipc_trace_port_rx(struct iosm_imem *ipc_imem, struct sk_buff *skb)
  19. {
  20. struct iosm_trace *ipc_trace = ipc_imem->trace;
  21. if (ipc_trace->ipc_rchan)
  22. relay_write(ipc_trace->ipc_rchan, skb->data, skb->len);
  23. dev_kfree_skb(skb);
  24. }
  25. /* Creates relay file in debugfs. */
  26. static struct dentry *
  27. ipc_trace_create_buf_file_handler(const char *filename,
  28. struct dentry *parent,
  29. umode_t mode,
  30. struct rchan_buf *buf,
  31. int *is_global)
  32. {
  33. *is_global = 1;
  34. return debugfs_create_file(filename, mode, parent, buf,
  35. &relay_file_operations);
  36. }
  37. /* Removes relay file from debugfs. */
  38. static int ipc_trace_remove_buf_file_handler(struct dentry *dentry)
  39. {
  40. debugfs_remove(dentry);
  41. return 0;
  42. }
  43. static int ipc_trace_subbuf_start_handler(struct rchan_buf *buf, void *subbuf,
  44. void *prev_subbuf)
  45. {
  46. if (relay_buf_full(buf)) {
  47. pr_err_ratelimited("Relay_buf full dropping traces");
  48. return 0;
  49. }
  50. return 1;
  51. }
  52. /* Relay interface callbacks */
  53. static struct rchan_callbacks relay_callbacks = {
  54. .subbuf_start = ipc_trace_subbuf_start_handler,
  55. .create_buf_file = ipc_trace_create_buf_file_handler,
  56. .remove_buf_file = ipc_trace_remove_buf_file_handler,
  57. };
  58. /* Copy the trace control mode to user buffer */
  59. static ssize_t ipc_trace_ctrl_file_read(struct file *filp, char __user *buffer,
  60. size_t count, loff_t *ppos)
  61. {
  62. struct iosm_trace *ipc_trace = filp->private_data;
  63. char buf[16];
  64. int len;
  65. mutex_lock(&ipc_trace->trc_mutex);
  66. len = snprintf(buf, sizeof(buf), "%d\n", ipc_trace->mode);
  67. mutex_unlock(&ipc_trace->trc_mutex);
  68. return simple_read_from_buffer(buffer, count, ppos, buf, len);
  69. }
  70. /* Open and close the trace channel depending on user input */
  71. static ssize_t ipc_trace_ctrl_file_write(struct file *filp,
  72. const char __user *buffer,
  73. size_t count, loff_t *ppos)
  74. {
  75. struct iosm_trace *ipc_trace = filp->private_data;
  76. unsigned long val;
  77. int ret;
  78. ret = kstrtoul_from_user(buffer, count, 10, &val);
  79. if (ret)
  80. return ret;
  81. mutex_lock(&ipc_trace->trc_mutex);
  82. if (val == TRACE_ENABLE && ipc_trace->mode != TRACE_ENABLE) {
  83. ipc_trace->channel = ipc_imem_sys_port_open(ipc_trace->ipc_imem,
  84. ipc_trace->chl_id,
  85. IPC_HP_CDEV_OPEN);
  86. if (!ipc_trace->channel) {
  87. ret = -EIO;
  88. goto unlock;
  89. }
  90. ipc_trace->mode = TRACE_ENABLE;
  91. } else if (val == TRACE_DISABLE && ipc_trace->mode != TRACE_DISABLE) {
  92. ipc_trace->mode = TRACE_DISABLE;
  93. /* close trace channel */
  94. ipc_imem_sys_port_close(ipc_trace->ipc_imem,
  95. ipc_trace->channel);
  96. relay_flush(ipc_trace->ipc_rchan);
  97. }
  98. ret = count;
  99. unlock:
  100. mutex_unlock(&ipc_trace->trc_mutex);
  101. return ret;
  102. }
  103. static const struct file_operations ipc_trace_fops = {
  104. .open = simple_open,
  105. .write = ipc_trace_ctrl_file_write,
  106. .read = ipc_trace_ctrl_file_read,
  107. };
  108. /**
  109. * ipc_trace_init - Create trace interface & debugfs entries
  110. * @ipc_imem: Pointer to iosm_imem structure
  111. *
  112. * Returns: Pointer to trace instance on success else NULL
  113. */
  114. struct iosm_trace *ipc_trace_init(struct iosm_imem *ipc_imem)
  115. {
  116. struct ipc_chnl_cfg chnl_cfg = { 0 };
  117. struct iosm_trace *ipc_trace;
  118. ipc_chnl_cfg_get(&chnl_cfg, IPC_MEM_CTRL_CHL_ID_3);
  119. ipc_imem_channel_init(ipc_imem, IPC_CTYPE_CTRL, chnl_cfg,
  120. IRQ_MOD_OFF);
  121. ipc_trace = kzalloc_obj(*ipc_trace);
  122. if (!ipc_trace)
  123. return NULL;
  124. ipc_trace->mode = TRACE_DISABLE;
  125. ipc_trace->dev = ipc_imem->dev;
  126. ipc_trace->ipc_imem = ipc_imem;
  127. ipc_trace->chl_id = IPC_MEM_CTRL_CHL_ID_3;
  128. mutex_init(&ipc_trace->trc_mutex);
  129. ipc_trace->ctrl_file = debugfs_create_file(IOSM_TRC_DEBUGFS_TRACE_CTRL,
  130. IOSM_TRC_FILE_PERM,
  131. ipc_imem->debugfs_dir,
  132. ipc_trace, &ipc_trace_fops);
  133. ipc_trace->ipc_rchan = relay_open(IOSM_TRC_DEBUGFS_TRACE,
  134. ipc_imem->debugfs_dir,
  135. IOSM_TRC_SUB_BUFF_SIZE,
  136. IOSM_TRC_N_SUB_BUFF,
  137. &relay_callbacks, NULL);
  138. return ipc_trace;
  139. }
  140. /**
  141. * ipc_trace_deinit - Closing relayfs, removing debugfs entries
  142. * @ipc_trace: Pointer to the iosm_trace data struct
  143. */
  144. void ipc_trace_deinit(struct iosm_trace *ipc_trace)
  145. {
  146. if (!ipc_trace)
  147. return;
  148. debugfs_remove(ipc_trace->ctrl_file);
  149. relay_close(ipc_trace->ipc_rchan);
  150. mutex_destroy(&ipc_trace->trc_mutex);
  151. kfree(ipc_trace);
  152. }