ath_procfs.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. /*
  2. * Copyright (c) 2013-2014, 2016 The Linux Foundation. All rights reserved.
  3. *
  4. * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
  5. *
  6. *
  7. * Permission to use, copy, modify, and/or distribute this software for
  8. * any purpose with or without fee is hereby granted, provided that the
  9. * above copyright notice and this permission notice appear in all
  10. * copies.
  11. *
  12. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  13. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  14. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  15. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  16. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  17. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  18. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  19. * PERFORMANCE OF THIS SOFTWARE.
  20. */
  21. /*
  22. * This file was originally distributed by Qualcomm Atheros, Inc.
  23. * under proprietary terms before Copyright ownership was assigned
  24. * to the Linux Foundation.
  25. */
  26. #if defined(CONFIG_ATH_PROCFS_DIAG_SUPPORT)
  27. #include <linux/module.h> /* Specifically, a module */
  28. #include <linux/kernel.h> /* We're doing kernel work */
  29. #include <linux/version.h> /* We're doing kernel work */
  30. #include <linux/proc_fs.h> /* Necessary because we use the proc fs */
  31. #include <asm/uaccess.h> /* for copy_from_user */
  32. #include "hif.h"
  33. #include "hif_main.h"
  34. #if defined(HIF_USB)
  35. #include "if_usb.h"
  36. #endif
  37. #if defined(HIF_SDIO)
  38. #include "if_sdio.h"
  39. #endif
  40. #include "hif_debug.h"
  41. #include "pld_common.h"
  42. #define PROCFS_NAME "athdiagpfs"
  43. #ifdef MULTI_IF_NAME
  44. #define PROCFS_DIR "cld" MULTI_IF_NAME
  45. #else
  46. #define PROCFS_DIR "cld"
  47. #endif
  48. /**
  49. * This structure hold information about the /proc file
  50. *
  51. */
  52. static struct proc_dir_entry *proc_file, *proc_dir;
  53. static void *get_hif_hdl_from_file(struct file *file)
  54. {
  55. struct hif_opaque_softc *scn;
  56. scn = (struct hif_opaque_softc *)PDE_DATA(file_inode(file));
  57. return (void *)scn;
  58. }
  59. static ssize_t ath_procfs_diag_read(struct file *file, char __user *buf,
  60. size_t count, loff_t *pos)
  61. {
  62. hif_handle_t hif_hdl;
  63. int rv;
  64. uint8_t *read_buffer = NULL;
  65. struct hif_softc *scn;
  66. uint32_t offset = 0, memtype = 0;
  67. read_buffer = qdf_mem_malloc(count);
  68. if (NULL == read_buffer) {
  69. HIF_ERROR("%s: cdf_mem_alloc failed", __func__);
  70. return -ENOMEM;
  71. }
  72. hif_hdl = get_hif_hdl_from_file(file);
  73. HIF_DBG("rd buff 0x%p cnt %zu offset 0x%x buf 0x%p",
  74. read_buffer, count, (int)*pos, buf);
  75. scn = HIF_GET_SOFTC(hif_hdl);
  76. if (scn->bus_type == QDF_BUS_TYPE_SNOC) {
  77. memtype = ((uint32_t)(*pos) & 0xff000000) >> 24;
  78. offset = (uint32_t)(*pos) & 0xffffff;
  79. HIF_TRACE("%s: offset 0x%x memtype 0x%x, datalen %zu\n",
  80. __func__, offset, memtype, count);
  81. rv = pld_athdiag_read(scn->qdf_dev->dev,
  82. offset, memtype, count,
  83. (uint8_t *)read_buffer);
  84. goto out;
  85. }
  86. if ((count == 4) && ((((uint32_t) (*pos)) & 3) == 0)) {
  87. /* reading a word? */
  88. rv = hif_diag_read_access(hif_hdl, (uint32_t)(*pos),
  89. (uint32_t *)read_buffer);
  90. } else {
  91. rv = hif_diag_read_mem(hif_hdl, (uint32_t)(*pos),
  92. (uint8_t *)read_buffer, count);
  93. }
  94. out:
  95. if (copy_to_user(buf, read_buffer, count)) {
  96. qdf_mem_free(read_buffer);
  97. HIF_ERROR("%s: copy_to_user error in /proc/%s",
  98. __func__, PROCFS_NAME);
  99. return -EFAULT;
  100. } else
  101. qdf_mem_free(read_buffer);
  102. if (rv == 0) {
  103. return count;
  104. } else {
  105. return -EIO;
  106. }
  107. }
  108. static ssize_t ath_procfs_diag_write(struct file *file,
  109. const char __user *buf,
  110. size_t count, loff_t *pos)
  111. {
  112. hif_handle_t hif_hdl;
  113. int rv;
  114. uint8_t *write_buffer = NULL;
  115. struct hif_softc *scn;
  116. uint32_t offset = 0, memtype = 0;
  117. write_buffer = qdf_mem_malloc(count);
  118. if (NULL == write_buffer) {
  119. HIF_ERROR("%s: cdf_mem_alloc failed", __func__);
  120. return -ENOMEM;
  121. }
  122. if (copy_from_user(write_buffer, buf, count)) {
  123. qdf_mem_free(write_buffer);
  124. HIF_ERROR("%s: copy_to_user error in /proc/%s",
  125. __func__, PROCFS_NAME);
  126. return -EFAULT;
  127. }
  128. hif_hdl = get_hif_hdl_from_file(file);
  129. HIF_DBG("wr buff 0x%p buf 0x%p cnt %zu offset 0x%x value 0x%x",
  130. write_buffer, buf, count,
  131. (int)*pos, *((uint32_t *) write_buffer));
  132. scn = HIF_GET_SOFTC(hif_hdl);
  133. if (scn->bus_type == QDF_BUS_TYPE_SNOC) {
  134. memtype = ((uint32_t)(*pos) & 0xff000000) >> 24;
  135. offset = (uint32_t)(*pos) & 0xffffff;
  136. HIF_TRACE("%s: offset 0x%x memtype 0x%x, datalen %zu\n",
  137. __func__, offset, memtype, count);
  138. rv = pld_athdiag_write(scn->qdf_dev->dev,
  139. offset, memtype, count,
  140. (uint8_t *)write_buffer);
  141. goto out;
  142. }
  143. if ((count == 4) && ((((uint32_t) (*pos)) & 3) == 0)) {
  144. /* reading a word? */
  145. uint32_t value = *((uint32_t *)write_buffer);
  146. rv = hif_diag_write_access(hif_hdl, (uint32_t)(*pos), value);
  147. } else {
  148. rv = hif_diag_write_mem(hif_hdl, (uint32_t)(*pos),
  149. (uint8_t *)write_buffer, count);
  150. }
  151. out:
  152. qdf_mem_free(write_buffer);
  153. if (rv == 0) {
  154. return count;
  155. } else {
  156. return -EIO;
  157. }
  158. }
  159. static const struct file_operations athdiag_fops = {
  160. .read = ath_procfs_diag_read,
  161. .write = ath_procfs_diag_write,
  162. };
  163. /**
  164. *This function is called when the module is loaded
  165. *
  166. */
  167. int athdiag_procfs_init(void *scn)
  168. {
  169. proc_dir = proc_mkdir(PROCFS_DIR, NULL);
  170. if (proc_dir == NULL) {
  171. remove_proc_entry(PROCFS_DIR, NULL);
  172. HIF_ERROR("%s: Error: Could not initialize /proc/%s",
  173. __func__, PROCFS_DIR);
  174. return -ENOMEM;
  175. }
  176. proc_file = proc_create_data(PROCFS_NAME,
  177. S_IRUSR | S_IWUSR, proc_dir,
  178. &athdiag_fops, (void *)scn);
  179. if (proc_file == NULL) {
  180. remove_proc_entry(PROCFS_NAME, proc_dir);
  181. HIF_ERROR("%s: Could not initialize /proc/%s",
  182. __func__, PROCFS_NAME);
  183. return -ENOMEM;
  184. }
  185. HIF_DBG("/proc/%s/%s created", PROCFS_DIR, PROCFS_NAME);
  186. return 0; /* everything is ok */
  187. }
  188. /**
  189. *This function is called when the module is unloaded
  190. *
  191. */
  192. void athdiag_procfs_remove(void)
  193. {
  194. if (proc_dir != NULL) {
  195. remove_proc_entry(PROCFS_NAME, proc_dir);
  196. HIF_DBG("/proc/%s/%s removed", PROCFS_DIR, PROCFS_NAME);
  197. remove_proc_entry(PROCFS_DIR, NULL);
  198. HIF_DBG("/proc/%s removed", PROCFS_DIR);
  199. proc_dir = NULL;
  200. }
  201. }
  202. #else
  203. int athdiag_procfs_init(void *scn)
  204. {
  205. return 0;
  206. }
  207. void athdiag_procfs_remove(void) {}
  208. #endif