xref: /linux/sound/soc/codecs/wcd-common.c (revision 309e94a64b61d1b18a0d0d83de4bf2d2582708ce)
1 // SPDX-License-Identifier: GPL-2.0-only
2 // Copyright (c) 2025, Qualcomm Technologies, Inc. and/or its subsidiaries.
3 
4 #include <linux/export.h>
5 #include <linux/module.h>
6 #include <linux/init.h>
7 #include <linux/device.h>
8 #include <linux/of.h>
9 #include <linux/printk.h>
10 #include <linux/component.h>
11 #include <linux/pm_runtime.h>
12 #include <linux/soundwire/sdw.h>
13 #include <linux/soundwire/sdw_type.h>
14 #include <linux/regmap.h>
15 
16 #include "wcd-common.h"
17 
18 #define WCD_MIN_MICBIAS_MV	1000
19 #define WCD_DEF_MICBIAS_MV	1800
20 #define WCD_MAX_MICBIAS_MV	2850
21 
22 #define SWRS_SCP_HOST_CLK_DIV2_CTL_BANK(m) (0xE0 + 0x10 * (m))
23 
24 int wcd_get_micb_vout_ctl_val(struct device *dev, u32 micb_mv)
25 {
26 	/* min micbias voltage is 1V and maximum is 2.85V */
27 	if (micb_mv < WCD_MIN_MICBIAS_MV || micb_mv > WCD_MAX_MICBIAS_MV) {
28 		dev_err(dev, "Unsupported micbias voltage (%u mV)\n", micb_mv);
29 		return -EINVAL;
30 	}
31 
32 	return (micb_mv - WCD_MIN_MICBIAS_MV) / 50;
33 }
34 EXPORT_SYMBOL_GPL(wcd_get_micb_vout_ctl_val);
35 
36 static int wcd_get_micbias_val(struct device *dev, int micb_num, u32 *micb_mv)
37 {
38 	char micbias[64];
39 	int mv;
40 
41 	sprintf(micbias, "qcom,micbias%d-microvolt", micb_num);
42 
43 	if (of_property_read_u32(dev->of_node, micbias, &mv)) {
44 		dev_err(dev, "%s value not found, using default\n", micbias);
45 		mv = WCD_DEF_MICBIAS_MV;
46 	} else {
47 		/* convert it to milli volts */
48 		mv = mv/1000;
49 	}
50 	if (micb_mv)
51 		*micb_mv = mv;
52 
53 	mv = wcd_get_micb_vout_ctl_val(dev, mv);
54 	if (mv < 0) {
55 		dev_err(dev, "Unsupported %s voltage (%d mV), falling back to default (%d mV)\n",
56 				micbias, mv, WCD_DEF_MICBIAS_MV);
57 		return wcd_get_micb_vout_ctl_val(dev, WCD_DEF_MICBIAS_MV);
58 	}
59 
60 	return mv;
61 }
62 
63 int wcd_dt_parse_micbias_info(struct wcd_common *common)
64 {
65 	int i;
66 
67 	for (i = 0; i < common->max_bias; i++) {
68 		common->micb_vout[i] = wcd_get_micbias_val(common->dev, i + 1, &common->micb_mv[i]);
69 		if (common->micb_vout[i] < 0)
70 			return -EINVAL;
71 	}
72 
73 	return 0;
74 }
75 EXPORT_SYMBOL_GPL(wcd_dt_parse_micbias_info);
76 
77 static int wcd_sdw_component_bind(struct device *dev, struct device *master, void *data)
78 {
79 	pm_runtime_set_autosuspend_delay(dev, 3000);
80 	pm_runtime_use_autosuspend(dev);
81 	pm_runtime_mark_last_busy(dev);
82 	pm_runtime_set_active(dev);
83 	pm_runtime_enable(dev);
84 
85 	return 0;
86 }
87 
88 static void wcd_sdw_component_unbind(struct device *dev, struct device *master, void *data)
89 {
90 	pm_runtime_disable(dev);
91 	pm_runtime_set_suspended(dev);
92 	pm_runtime_dont_use_autosuspend(dev);
93 }
94 
95 const struct component_ops wcd_sdw_component_ops = {
96 	.bind = wcd_sdw_component_bind,
97 	.unbind = wcd_sdw_component_unbind,
98 };
99 EXPORT_SYMBOL_GPL(wcd_sdw_component_ops);
100 
101 int wcd_update_status(struct sdw_slave *slave, enum sdw_slave_status status)
102 {
103 	struct regmap *regmap = dev_get_regmap(&slave->dev, NULL);
104 
105 	if (regmap && status == SDW_SLAVE_ATTACHED) {
106 		/* Write out any cached changes that happened between probe and attach */
107 		regcache_cache_only(regmap, false);
108 		return regcache_sync(regmap);
109 	}
110 
111 	return 0;
112 }
113 EXPORT_SYMBOL_GPL(wcd_update_status);
114 
115 int wcd_bus_config(struct sdw_slave *slave, struct sdw_bus_params *params)
116 {
117 	sdw_write(slave, SWRS_SCP_HOST_CLK_DIV2_CTL_BANK(params->next_bank), 0x01);
118 
119 	return 0;
120 }
121 EXPORT_SYMBOL_GPL(wcd_bus_config);
122 
123 int wcd_interrupt_callback(struct sdw_slave *slave, struct irq_domain *slave_irq,
124 		unsigned int wcd_intr_status0, unsigned int wcd_intr_status1,
125 		unsigned int wcd_intr_status2)
126 {
127 	struct regmap *regmap = dev_get_regmap(&slave->dev, NULL);
128 	u32 sts1, sts2, sts3;
129 
130 	do {
131 		handle_nested_irq(irq_find_mapping(slave_irq, 0));
132 		regmap_read(regmap, wcd_intr_status0, &sts1);
133 		regmap_read(regmap, wcd_intr_status1, &sts2);
134 		regmap_read(regmap, wcd_intr_status2, &sts3);
135 
136 	} while (sts1 || sts2 || sts3);
137 
138 	return IRQ_HANDLED;
139 }
140 EXPORT_SYMBOL_GPL(wcd_interrupt_callback);
141 
142 MODULE_DESCRIPTION("Common Qualcomm WCD Codec helpers driver");
143 MODULE_LICENSE("GPL");
144