xref: /linux/sound/soc/codecs/rt-sdw-common.c (revision 071b34dcf71523a559b6c39f5d21a268a9531b50)
1 // SPDX-License-Identifier: GPL-2.0-only
2 //
3 // rt-sdw-common.c
4 //
5 // Copyright(c) 2024 Realtek Semiconductor Corp.
6 //
7 
8 /*
9  * This file defines common functions used with Realtek soundwire codecs.
10  */
11 
12 #include <linux/module.h>
13 #include <linux/regmap.h>
14 #include <linux/bitops.h>
15 #include <linux/soundwire/sdw_registers.h>
16 #include <sound/jack.h>
17 
18 #include "rt-sdw-common.h"
19 
20 /**
21  * rt_sdca_index_write - Write a value to Realtek defined register.
22  *
23  * @map: map for setting.
24  * @nid: Realtek-defined ID.
25  * @reg: register.
26  * @value: value.
27  *
28  * A value of zero will be returned on success, a negative errno will
29  * be returned in error cases.
30  */
31 int rt_sdca_index_write(struct regmap *map, unsigned int nid,
32 	unsigned int reg, unsigned int value)
33 {
34 	unsigned int addr = (nid << 20) | reg;
35 	int ret;
36 
37 	ret = regmap_write(map, addr, value);
38 	if (ret < 0)
39 		pr_err("Failed to set value: %06x <= %04x ret=%d\n",
40 			addr, value, ret);
41 
42 	return ret;
43 }
44 EXPORT_SYMBOL_GPL(rt_sdca_index_write);
45 
46 /**
47  * rt_sdca_index_read - Read value from Realtek defined register.
48  *
49  * @map: map for setting.
50  * @nid: Realtek-defined ID.
51  * @reg: register.
52  * @value: value.
53  *
54  * A value of zero will be returned on success, a negative errno will
55  * be returned in error cases.
56  */
57 int rt_sdca_index_read(struct regmap *map, unsigned int nid,
58 	unsigned int reg, unsigned int *value)
59 {
60 	unsigned int addr = (nid << 20) | reg;
61 	int ret;
62 
63 	ret = regmap_read(map, addr, value);
64 	if (ret < 0)
65 		pr_err("Failed to get value: %06x => %04x ret=%d\n",
66 			addr, *value, ret);
67 
68 	return ret;
69 }
70 EXPORT_SYMBOL_GPL(rt_sdca_index_read);
71 
72 /**
73  * rt_sdca_index_update_bits - Update value on Realtek defined register.
74  *
75  * @map: map for setting.
76  * @nid: Realtek-defined ID.
77  * @reg: register.
78  * @mask: Bitmask to change
79  * @val: New value for bitmask
80  *
81  * A value of zero will be returned on success, a negative errno will
82  * be returned in error cases.
83  */
84 
85 int rt_sdca_index_update_bits(struct regmap *map,
86 	unsigned int nid, unsigned int reg, unsigned int mask, unsigned int val)
87 {
88 	unsigned int tmp;
89 	int ret;
90 
91 	ret = rt_sdca_index_read(map, nid, reg, &tmp);
92 	if (ret < 0)
93 		return ret;
94 
95 	set_mask_bits(&tmp, mask, val);
96 	return rt_sdca_index_write(map, nid, reg, tmp);
97 }
98 EXPORT_SYMBOL_GPL(rt_sdca_index_update_bits);
99 
100 /**
101  * rt_sdca_btn_type - Decision of button type.
102  *
103  * @buffer: UMP message buffer.
104  *
105  * A button type will be returned regarding to buffer,
106  * it returns zero if buffer cannot be recognized.
107  */
108 int rt_sdca_btn_type(unsigned char *buffer)
109 {
110 	u8 btn_type = 0;
111 	int ret = 0;
112 
113 	btn_type |= buffer[0] & 0xf;
114 	btn_type |= (buffer[0] >> 4) & 0xf;
115 	btn_type |= buffer[1] & 0xf;
116 	btn_type |= (buffer[1] >> 4) & 0xf;
117 
118 	if (btn_type & BIT(0))
119 		ret |= SND_JACK_BTN_2;
120 	if (btn_type & BIT(1))
121 		ret |= SND_JACK_BTN_3;
122 	if (btn_type & BIT(2))
123 		ret |= SND_JACK_BTN_0;
124 	if (btn_type & BIT(3))
125 		ret |= SND_JACK_BTN_1;
126 
127 	return ret;
128 }
129 EXPORT_SYMBOL_GPL(rt_sdca_btn_type);
130 
131 /**
132  * rt_sdca_headset_detect - Headset jack type detection.
133  *
134  * @map: map for setting.
135  * @entity_id: SDCA entity ID.
136  *
137  * A headset jack type will be returned, a negative errno will
138  * be returned in error cases.
139  */
140 int rt_sdca_headset_detect(struct regmap *map, unsigned int entity_id)
141 {
142 	unsigned int det_mode, jack_type;
143 	int ret;
144 
145 	/* get detected_mode */
146 	ret = regmap_read(map, SDW_SDCA_CTL(SDCA_NUM_JACK_CODEC, entity_id,
147 			RT_SDCA_CTL_DETECTED_MODE, 0), &det_mode);
148 
149 	if (ret < 0)
150 		goto io_error;
151 
152 	switch (det_mode) {
153 	case 0x03:
154 		jack_type = SND_JACK_HEADPHONE;
155 		break;
156 	case 0x05:
157 		jack_type = SND_JACK_HEADSET;
158 		break;
159 	default:
160 		jack_type = 0;
161 		break;
162 	}
163 
164 	/* write selected_mode */
165 	if (det_mode) {
166 		ret = regmap_write(map, SDW_SDCA_CTL(SDCA_NUM_JACK_CODEC, entity_id,
167 				RT_SDCA_CTL_SELECTED_MODE, 0), det_mode);
168 		if (ret < 0)
169 			goto io_error;
170 	}
171 
172 	return jack_type;
173 
174 io_error:
175 	pr_err_ratelimited("IO error in %s, ret %d\n", __func__, ret);
176 	return ret;
177 }
178 EXPORT_SYMBOL_GPL(rt_sdca_headset_detect);
179 
180 /**
181  * rt_sdca_button_detect - Read UMP message and decide button type.
182  *
183  * @map: map for setting.
184  * @entity_id: SDCA entity ID.
185  * @hid_buf_addr: HID buffer address.
186  * @hid_id: Report ID for HID.
187  *
188  * A button type will be returned regarding to buffer,
189  * it returns zero if buffer cannot be recognized.
190  */
191 int rt_sdca_button_detect(struct regmap *map, unsigned int entity_id,
192 	unsigned int hid_buf_addr, unsigned int hid_id)
193 {
194 	unsigned int btn_type = 0, offset, idx, val, owner;
195 	unsigned char buf[3];
196 	int ret;
197 
198 	/* get current UMP message owner */
199 	ret = regmap_read(map, SDW_SDCA_CTL(SDCA_NUM_HID, entity_id,
200 			RT_SDCA_CTL_HIDTX_CURRENT_OWNER, 0), &owner);
201 	if (ret < 0)
202 		return 0;
203 
204 	/* if owner is device then there is no button event from device */
205 	if (owner == 1)
206 		return 0;
207 
208 	/* read UMP message offset */
209 	ret = regmap_read(map, SDW_SDCA_CTL(SDCA_NUM_HID, entity_id,
210 			RT_SDCA_CTL_HIDTX_MESSAGE_OFFSET, 0), &offset);
211 	if (ret < 0)
212 		goto _end_btn_det_;
213 
214 	for (idx = 0; idx < sizeof(buf); idx++) {
215 		ret = regmap_read(map, hid_buf_addr + offset + idx, &val);
216 		if (ret < 0)
217 			goto _end_btn_det_;
218 		buf[idx] = val & 0xff;
219 	}
220 	/* Report ID for HID */
221 	if (buf[0] == hid_id)
222 		btn_type = rt_sdca_btn_type(&buf[1]);
223 
224 _end_btn_det_:
225 	/* Host is owner, so set back to device */
226 	if (owner == 0)
227 		/* set owner to device */
228 		regmap_write(map,
229 			SDW_SDCA_CTL(SDCA_NUM_HID, entity_id,
230 				RT_SDCA_CTL_HIDTX_CURRENT_OWNER, 0), 0x01);
231 
232 	return btn_type;
233 }
234 EXPORT_SYMBOL_GPL(rt_sdca_button_detect);
235 
236 MODULE_DESCRIPTION("Realtek soundwire common functions");
237 MODULE_AUTHOR("jack yu <jack.yu@realtek.com>");
238 MODULE_LICENSE("GPL");
239