q6afecal-hwdep.c 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2015, 2017, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/kernel.h>
  6. #include <linux/platform_device.h>
  7. #include <linux/ioctl.h>
  8. #include <sound/hwdep.h>
  9. #include <sound/msmcal-hwdep.h>
  10. #include <sound/soc.h>
  11. #include "q6afecal-hwdep.h"
  12. const int cal_size_info[Q6AFE_MAX_CAL] = {
  13. [Q6AFE_VAD_CORE_CAL] = 132,
  14. };
  15. const char *cal_name_info[Q6AFE_MAX_CAL] = {
  16. [Q6AFE_VAD_CORE_CAL] = "vad_core",
  17. };
  18. #define AFE_HW_NAME_LENGTH 40
  19. /*
  20. * q6afecal_get_fw_cal -
  21. * To get calibration from AFE HW dependent node
  22. *
  23. * @fw_data: pointer to firmware data
  24. * type: AFE calibration type
  25. *
  26. */
  27. struct firmware_cal *q6afecal_get_fw_cal(struct afe_fw_info *fw_data,
  28. enum q6afe_cal_type type)
  29. {
  30. if (!fw_data) {
  31. pr_err("%s: fw_data is NULL\n", __func__);
  32. return NULL;
  33. }
  34. if (type >= Q6AFE_MAX_CAL ||
  35. type < Q6AFE_MIN_CAL) {
  36. pr_err("%s: wrong cal type sent %d\n", __func__, type);
  37. return NULL;
  38. }
  39. mutex_lock(&fw_data->lock);
  40. if (!test_bit(Q6AFECAL_RECEIVED,
  41. &fw_data->q6afecal_state[type])) {
  42. pr_err("%s: cal not sent by userspace %d\n",
  43. __func__, type);
  44. mutex_unlock(&fw_data->lock);
  45. return NULL;
  46. }
  47. set_bit(Q6AFECAL_INITIALISED, &fw_data->q6afecal_state[type]);
  48. mutex_unlock(&fw_data->lock);
  49. return fw_data->fw[type];
  50. }
  51. EXPORT_SYMBOL(q6afecal_get_fw_cal);
  52. static int q6afecal_hwdep_ioctl_shared(struct snd_hwdep *hw,
  53. struct q6afecal_ioctl_buffer fw_user)
  54. {
  55. struct afe_fw_info *fw_data = hw->private_data;
  56. struct firmware_cal **fw = fw_data->fw;
  57. void *data;
  58. if (!test_bit(fw_user.cal_type, fw_data->cal_bit)) {
  59. pr_err("%s: q6afe didn't set this %d!!\n",
  60. __func__, fw_user.cal_type);
  61. return -EFAULT;
  62. }
  63. if (fw_user.cal_type >= Q6AFE_MAX_CAL ||
  64. fw_user.cal_type < Q6AFE_MIN_CAL) {
  65. pr_err("%s: wrong cal type sent %d\n",
  66. __func__, fw_user.cal_type);
  67. return -EFAULT;
  68. }
  69. if (fw_user.size > cal_size_info[fw_user.cal_type] ||
  70. fw_user.size <= 0) {
  71. pr_err("%s: incorrect firmware size %d for %s\n",
  72. __func__, fw_user.size,
  73. cal_name_info[fw_user.cal_type]);
  74. return -EFAULT;
  75. }
  76. data = fw[fw_user.cal_type]->data;
  77. if (copy_from_user(data, fw_user.buffer, fw_user.size))
  78. return -EFAULT;
  79. fw[fw_user.cal_type]->size = fw_user.size;
  80. mutex_lock(&fw_data->lock);
  81. set_bit(Q6AFECAL_RECEIVED, &fw_data->q6afecal_state[fw_user.cal_type]);
  82. mutex_unlock(&fw_data->lock);
  83. return 0;
  84. }
  85. #ifdef CONFIG_COMPAT
  86. struct q6afecal_ioctl_buffer32 {
  87. u32 size;
  88. compat_uptr_t buffer;
  89. enum q6afe_cal_type cal_type;
  90. };
  91. enum {
  92. SNDRV_CTL_IOCTL_HWDEP_CAL_TYPE32 =
  93. _IOW('U', 0x1, struct q6afecal_ioctl_buffer32),
  94. };
  95. static int q6afecal_hwdep_ioctl_compat(struct snd_hwdep *hw, struct file *file,
  96. unsigned int cmd, unsigned long arg)
  97. {
  98. struct q6afecal_ioctl_buffer __user *argp = (void __user *)arg;
  99. struct q6afecal_ioctl_buffer32 fw_user32;
  100. struct q6afecal_ioctl_buffer fw_user_compat;
  101. if (cmd != SNDRV_CTL_IOCTL_HWDEP_CAL_TYPE32) {
  102. pr_err("%s: wrong ioctl command sent %u!\n", __func__, cmd);
  103. return -ENOIOCTLCMD;
  104. }
  105. if (copy_from_user(&fw_user32, argp, sizeof(fw_user32))) {
  106. pr_err("%s: failed to copy\n", __func__);
  107. return -EFAULT;
  108. }
  109. fw_user_compat.size = fw_user32.size;
  110. fw_user_compat.buffer = compat_ptr(fw_user32.buffer);
  111. fw_user_compat.cal_type = fw_user32.cal_type;
  112. return q6afecal_hwdep_ioctl_shared(hw, fw_user_compat);
  113. }
  114. #else
  115. #define q6afecal_hwdep_ioctl_compat NULL
  116. #endif
  117. static int q6afecal_hwdep_ioctl(struct snd_hwdep *hw, struct file *file,
  118. unsigned int cmd, unsigned long arg)
  119. {
  120. struct q6afecal_ioctl_buffer __user *argp = (void __user *)arg;
  121. struct q6afecal_ioctl_buffer fw_user;
  122. if (cmd != SNDRV_IOCTL_HWDEP_VAD_CAL_TYPE) {
  123. pr_err("%s: wrong ioctl command sent %d!\n", __func__, cmd);
  124. return -ENOIOCTLCMD;
  125. }
  126. if (copy_from_user(&fw_user, argp, sizeof(fw_user))) {
  127. pr_err("%s: failed to copy\n", __func__);
  128. return -EFAULT;
  129. }
  130. return q6afecal_hwdep_ioctl_shared(hw, fw_user);
  131. }
  132. static int q6afecal_hwdep_release(struct snd_hwdep *hw, struct file *file)
  133. {
  134. struct afe_fw_info *fw_data = hw->private_data;
  135. mutex_lock(&fw_data->lock);
  136. /* clear all the calibrations */
  137. memset(fw_data->q6afecal_state, 0,
  138. sizeof(fw_data->q6afecal_state));
  139. mutex_unlock(&fw_data->lock);
  140. return 0;
  141. }
  142. /**
  143. * q6afe_cal_create_hwdep -
  144. * for creating HW dependent node for AFE
  145. *
  146. * @data: Pointer to hold fw data
  147. * @node: node type
  148. * @card: Pointer to sound card
  149. *
  150. */
  151. int q6afe_cal_create_hwdep(void *data, int node, void *card)
  152. {
  153. char hwname[AFE_HW_NAME_LENGTH];
  154. struct snd_hwdep *hwdep;
  155. struct firmware_cal **fw;
  156. struct afe_fw_info *fw_data = data;
  157. int err, cal_bit;
  158. if (!fw_data || !card) {
  159. pr_err("%s: Invalid parameters\n", __func__);
  160. return -EINVAL;
  161. }
  162. fw = fw_data->fw;
  163. snprintf(hwname, strlen("Q6AFE"), "Q6AFE");
  164. err = snd_hwdep_new(((struct snd_soc_card *)card)->snd_card,
  165. hwname, node, &hwdep);
  166. if (err < 0) {
  167. pr_err("%s: new hwdep for q6afe failed %d\n", __func__, err);
  168. return err;
  169. }
  170. snprintf(hwdep->name, strlen("Q6AFECAL"), "Q6AFECAL");
  171. hwdep->iface = SNDRV_HWDEP_IFACE_AUDIO_BE;
  172. hwdep->private_data = fw_data;
  173. hwdep->ops.ioctl_compat = q6afecal_hwdep_ioctl_compat;
  174. hwdep->ops.ioctl = q6afecal_hwdep_ioctl;
  175. hwdep->ops.release = q6afecal_hwdep_release;
  176. mutex_init(&fw_data->lock);
  177. for_each_set_bit(cal_bit, fw_data->cal_bit, Q6AFE_MAX_CAL) {
  178. set_bit(Q6AFECAL_UNINITIALISED,
  179. &fw_data->q6afecal_state[cal_bit]);
  180. fw[cal_bit] = kzalloc(sizeof *(fw[cal_bit]), GFP_KERNEL);
  181. if (!fw[cal_bit]) {
  182. pr_err("%s: no memory for %s cal\n",
  183. __func__, cal_name_info[cal_bit]);
  184. goto end;
  185. }
  186. }
  187. for_each_set_bit(cal_bit, fw_data->cal_bit, Q6AFE_MAX_CAL) {
  188. fw[cal_bit]->data = kzalloc(cal_size_info[cal_bit],
  189. GFP_KERNEL);
  190. if (!fw[cal_bit]->data)
  191. goto exit;
  192. set_bit(Q6AFECAL_INITIALISED,
  193. &fw_data->q6afecal_state[cal_bit]);
  194. }
  195. return 0;
  196. exit:
  197. for_each_set_bit(cal_bit, fw_data->cal_bit, Q6AFE_MAX_CAL) {
  198. kfree(fw[cal_bit]->data);
  199. fw[cal_bit]->data = NULL;
  200. }
  201. end:
  202. for_each_set_bit(cal_bit, fw_data->cal_bit, Q6AFE_MAX_CAL) {
  203. kfree(fw[cal_bit]);
  204. fw[cal_bit] = NULL;
  205. }
  206. return -ENOMEM;
  207. }
  208. EXPORT_SYMBOL(q6afe_cal_create_hwdep);