sdca_jack.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. // SPDX-License-Identifier: GPL-2.0
  2. // Copyright (C) 2025 Cirrus Logic, Inc. and
  3. // Cirrus Logic International Semiconductor Ltd.
  4. /*
  5. * The MIPI SDCA specification is available for public downloads at
  6. * https://www.mipi.org/mipi-sdca-v1-0-download
  7. */
  8. #include <linux/cleanup.h>
  9. #include <linux/device.h>
  10. #include <linux/dev_printk.h>
  11. #include <linux/soundwire/sdw.h>
  12. #include <linux/soundwire/sdw_registers.h>
  13. #include <linux/sprintf.h>
  14. #include <linux/regmap.h>
  15. #include <linux/rwsem.h>
  16. #include <sound/asound.h>
  17. #include <sound/control.h>
  18. #include <sound/jack.h>
  19. #include <sound/sdca.h>
  20. #include <sound/sdca_function.h>
  21. #include <sound/sdca_interrupts.h>
  22. #include <sound/sdca_jack.h>
  23. #include <sound/soc-component.h>
  24. #include <sound/soc-jack.h>
  25. #include <sound/soc.h>
  26. /**
  27. * sdca_jack_process - Process an SDCA jack event
  28. * @interrupt: SDCA interrupt structure
  29. *
  30. * Return: Zero on success or a negative error code.
  31. */
  32. int sdca_jack_process(struct sdca_interrupt *interrupt)
  33. {
  34. struct device *dev = interrupt->dev;
  35. struct snd_soc_component *component = interrupt->component;
  36. struct snd_soc_card *card = component->card;
  37. struct rw_semaphore *rwsem = &card->snd_card->controls_rwsem;
  38. struct jack_state *state = interrupt->priv;
  39. struct snd_kcontrol *kctl = state->kctl;
  40. struct snd_ctl_elem_value *ucontrol __free(kfree) = NULL;
  41. unsigned int reg, val;
  42. int ret;
  43. guard(rwsem_write)(rwsem);
  44. if (!kctl) {
  45. const char *name __free(kfree) = kasprintf(GFP_KERNEL, "%s %s",
  46. interrupt->entity->label,
  47. SDCA_CTL_SELECTED_MODE_NAME);
  48. if (!name)
  49. return -ENOMEM;
  50. kctl = snd_soc_component_get_kcontrol(component, name);
  51. if (!kctl)
  52. dev_dbg(dev, "control not found: %s\n", name);
  53. else
  54. state->kctl = kctl;
  55. }
  56. reg = SDW_SDCA_CTL(interrupt->function->desc->adr, interrupt->entity->id,
  57. interrupt->control->sel, 0);
  58. ret = regmap_read(interrupt->function_regmap, reg, &val);
  59. if (ret < 0) {
  60. dev_err(dev, "failed to read detected mode: %d\n", ret);
  61. return ret;
  62. }
  63. reg = SDW_SDCA_CTL(interrupt->function->desc->adr, interrupt->entity->id,
  64. SDCA_CTL_GE_SELECTED_MODE, 0);
  65. switch (val) {
  66. case SDCA_DETECTED_MODE_DETECTION_IN_PROGRESS:
  67. case SDCA_DETECTED_MODE_JACK_UNKNOWN:
  68. /*
  69. * Selected mode is not normally marked as volatile register
  70. * (RW), but here force a read from the hardware. If the
  71. * detected mode is unknown we need to see what the device
  72. * selected as a "safe" option.
  73. */
  74. regcache_drop_region(interrupt->function_regmap, reg, reg);
  75. ret = regmap_read(interrupt->function_regmap, reg, &val);
  76. if (ret) {
  77. dev_err(dev, "failed to re-check selected mode: %d\n", ret);
  78. return ret;
  79. }
  80. break;
  81. default:
  82. break;
  83. }
  84. dev_dbg(dev, "%s: %#x\n", interrupt->name, val);
  85. if (kctl) {
  86. struct soc_enum *soc_enum = (struct soc_enum *)kctl->private_value;
  87. ucontrol = kzalloc_obj(*ucontrol);
  88. if (!ucontrol)
  89. return -ENOMEM;
  90. ucontrol->value.enumerated.item[0] = snd_soc_enum_val_to_item(soc_enum, val);
  91. ret = snd_soc_dapm_put_enum_double(kctl, ucontrol);
  92. if (ret < 0) {
  93. dev_err(dev, "failed to update selected mode: %d\n", ret);
  94. return ret;
  95. }
  96. snd_ctl_notify(card->snd_card, SNDRV_CTL_EVENT_MASK_VALUE, &kctl->id);
  97. } else {
  98. ret = regmap_write(interrupt->function_regmap, reg, val);
  99. if (ret) {
  100. dev_err(dev, "failed to write selected mode: %d\n", ret);
  101. return ret;
  102. }
  103. }
  104. return sdca_jack_report(interrupt);
  105. }
  106. EXPORT_SYMBOL_NS_GPL(sdca_jack_process, "SND_SOC_SDCA");
  107. /**
  108. * sdca_jack_alloc_state - allocate state for a jack interrupt
  109. * @interrupt: SDCA interrupt structure.
  110. *
  111. * Return: Zero on success or a negative error code.
  112. */
  113. int sdca_jack_alloc_state(struct sdca_interrupt *interrupt)
  114. {
  115. struct device *dev = interrupt->dev;
  116. struct jack_state *jack_state;
  117. jack_state = devm_kzalloc(dev, sizeof(*jack_state), GFP_KERNEL);
  118. if (!jack_state)
  119. return -ENOMEM;
  120. interrupt->priv = jack_state;
  121. return 0;
  122. }
  123. EXPORT_SYMBOL_NS_GPL(sdca_jack_alloc_state, "SND_SOC_SDCA");
  124. /**
  125. * sdca_jack_set_jack - attach an ASoC jack to SDCA
  126. * @info: SDCA interrupt information.
  127. * @jack: ASoC jack to be attached.
  128. *
  129. * Return: Zero on success or a negative error code.
  130. */
  131. int sdca_jack_set_jack(struct sdca_interrupt_info *info, struct snd_soc_jack *jack)
  132. {
  133. int i, ret;
  134. guard(mutex)(&info->irq_lock);
  135. for (i = 0; i < SDCA_MAX_INTERRUPTS; i++) {
  136. struct sdca_interrupt *interrupt = &info->irqs[i];
  137. struct sdca_control *control = interrupt->control;
  138. struct sdca_entity *entity = interrupt->entity;
  139. struct jack_state *jack_state;
  140. if (!interrupt->irq)
  141. continue;
  142. switch (SDCA_CTL_TYPE(entity->type, control->sel)) {
  143. case SDCA_CTL_TYPE_S(GE, DETECTED_MODE):
  144. jack_state = interrupt->priv;
  145. jack_state->jack = jack;
  146. /* Report initial state in case IRQ was already handled */
  147. ret = sdca_jack_report(interrupt);
  148. if (ret)
  149. return ret;
  150. break;
  151. default:
  152. break;
  153. }
  154. }
  155. return 0;
  156. }
  157. EXPORT_SYMBOL_NS_GPL(sdca_jack_set_jack, "SND_SOC_SDCA");
  158. int sdca_jack_report(struct sdca_interrupt *interrupt)
  159. {
  160. struct jack_state *jack_state = interrupt->priv;
  161. struct sdca_control_range *range;
  162. enum sdca_terminal_type type;
  163. unsigned int report = 0;
  164. unsigned int reg, val;
  165. int ret;
  166. reg = SDW_SDCA_CTL(interrupt->function->desc->adr, interrupt->entity->id,
  167. SDCA_CTL_GE_SELECTED_MODE, 0);
  168. ret = regmap_read(interrupt->function_regmap, reg, &val);
  169. if (ret) {
  170. dev_err(interrupt->dev, "failed to read selected mode: %d\n", ret);
  171. return ret;
  172. }
  173. range = sdca_selector_find_range(interrupt->dev, interrupt->entity,
  174. SDCA_CTL_GE_SELECTED_MODE,
  175. SDCA_SELECTED_MODE_NCOLS, 0);
  176. if (!range)
  177. return -EINVAL;
  178. type = sdca_range_search(range, SDCA_SELECTED_MODE_INDEX,
  179. val, SDCA_SELECTED_MODE_TERM_TYPE);
  180. switch (type) {
  181. case SDCA_TERM_TYPE_LINEIN_STEREO:
  182. case SDCA_TERM_TYPE_LINEIN_FRONT_LR:
  183. case SDCA_TERM_TYPE_LINEIN_CENTER_LFE:
  184. case SDCA_TERM_TYPE_LINEIN_SURROUND_LR:
  185. case SDCA_TERM_TYPE_LINEIN_REAR_LR:
  186. report = SND_JACK_LINEIN;
  187. break;
  188. case SDCA_TERM_TYPE_LINEOUT_STEREO:
  189. case SDCA_TERM_TYPE_LINEOUT_FRONT_LR:
  190. case SDCA_TERM_TYPE_LINEOUT_CENTER_LFE:
  191. case SDCA_TERM_TYPE_LINEOUT_SURROUND_LR:
  192. case SDCA_TERM_TYPE_LINEOUT_REAR_LR:
  193. report = SND_JACK_LINEOUT;
  194. break;
  195. case SDCA_TERM_TYPE_MIC_JACK:
  196. report = SND_JACK_MICROPHONE;
  197. break;
  198. case SDCA_TERM_TYPE_HEADPHONE_JACK:
  199. report = SND_JACK_HEADPHONE;
  200. break;
  201. case SDCA_TERM_TYPE_HEADSET_JACK:
  202. report = SND_JACK_HEADSET;
  203. break;
  204. default:
  205. break;
  206. }
  207. snd_soc_jack_report(jack_state->jack, report, 0xFFFF);
  208. return 0;
  209. }
  210. EXPORT_SYMBOL_NS_GPL(sdca_jack_report, "SND_SOC_SDCA");